8 全 球 销量 适 上 下 万 册 的 系列 图 书 

e 连续 十 余年 打造 的 经 典 品牌 

四 直观 、 精 序 渐进 的 学 习 救 程 

画 掌握 关键 知识 的 最 佳 起 点 

uE “Read Less, Do More” (MiSs) 的 教学 理念 
u 以 示例 引导 读者 完成 最 常见 的 任务 


每 章 内 容 针 对 初学 者 精心 设计 ， 用 小 时 轻松 阅读 学 习 ， 


小 时 彻底 掌握 关键 知识 

随 书 附 赠 、 
每 章 助 你 轻松 完成 常见 任务 ， 。 国际- 入 
通过 提高 应 用 技能 ， 巩 固 所 学 知识 Linux 和 Mac OS X d 
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2.2 如 何 获 取 MySQL 

2.3 在 Linux/UNIX 上 安装 MySQL 
2.4 在 Mac OS X £28 MySOL 


2.5 在 Windows 上 安装 MySQL 
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内 容 拓 要 


PHP、MySQL 和 Apache 是 Web 应 用 开 有 的 强大 组 合 工 上 共 。 本 书 针对 
这 三 种 主流 工具 的 最 狐 版 本 ， 了 逐步 介绍 了 如 何 安装 、 配 置 和 使 用 这 些 工 
上 其 组 合 ， 并 通过 一 些 典 型 的 项 目 宁 例 ， 儿 助 读者 开发 出 功能 强大 的 Web 
应 用 。 


全 书 分 为 6 个 部 分 共 33 章 。 第 1 部 分 “基础 知识 ”， 包 括 第 1 章 到 第 4 
音 ， 引 领 读者 深入 了 解 、 安 装 和 配置 MySQL、Apache 和 PHP。 第 2 部 
分 "PHP 语言 结构 ”， 包 括 第 5 章 到 第 9 章 ， 讲 解 PHP 语 言 基础 ， 包 括 数组 
和 对 象 这 样 的 结构 化 元 隶 。 第 3 部 分 “ 深 入 编程 >， 包 括 第 10 章 到 第 14 
章 ， 介 绍 中 级 应 用 程序 开发 的 主题 ， 包 括 使 用 表单 和 文件 、 限 制 访 问 以 
及 完成 包含 某 个 专门 概念 的 小 项 目 。 第 4 部 分 “PHP 和 和 MySQL 整合 "， 包 
括 第 15 章 到 第 18 革 ， 介 绍 使 用 数据 库 的 一 般 方 法 。 第 5 部 分 “基本 项 
目 ”， 包 括 第 19 章 到 第 28 章 ， 介 绍 如 何 整 合 前 面 已 经 学 习 到 的 所 有 知 
识 ， 使 用 PHP 和 MySQL 执 行 一 个 特定 任务 。 第 6 部 分 “管理 和 优化 ”>， 包 
括 第 29 章 到 第 33 章 ， 介 绍 管理 和 优化 Apache 和 和 MySQL 的 方法 。 


本 书 内 容 人 全面、 讲解 详细 、 由 浅 入 深 、 实 例 丰 富 ， 而 且 考 虑 到 读者 
使 用 不 同 操作 系统 和 开发 环境 的 需求 。 本 书 可 作为 PHP、MySQL、 
Apache 人 初学 者 的 学 习 指 南 ， 也 可 作为 Web 开 发 技术 人 员 的 参考 用 书 。 
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欢迎 阅读 本 书 。 我 很 高 兴 地 告诉 你 ，PHP 语 言及 其 开发 者 社 群 和 用 
户 每 天 都 在 持续 增加 ， 因 此 ， 本 书 需 要 更 新 版 本 。 


本 书 前 一 个 版 本 介绍 的 PHP4， 其 “生命 尽头 ”终于 快 到 了 ， 在 
GoPHP5 活 动 的 帮助 下 ，Web 主 机 上 服务 提供 商 和 应 用 程序 开 友 者 迁移 其 
服务 和 代码 ， 逐 渐 放 弃 特 定 于 PHP 4 的 功能 和 代码 实践 ， 而 进入 PHP 5 的 
世界 。 这 是 一 个 更 快速 、 更 好 的 功能 集合 。 和 本 书 上 一 版 一 样 ， 本 版 中 
所 有 的 代码 都 是 基于 编写 本 书 的 时 候 PHP 最 新 的 可 用 版 本 (具体 来 说 碘 
是 PHP 5.4.0) 。 


你 可 能 已 经 听 说 了 PHP 6 或 者 已 经 看 到 一 些 图 书 宣传 自己 使 用 了 
PHP 6 作为 核心 语言 。 然 而 ，PHP 6 的 语言 版 本 还 没有 具体 化 ，PHP 6 计 
划 的 功能 ， 都 已 添加 到 了 PHP 5.3 和 PHP 5.4 中 。 因 此 ， 如 果 你 听 说 了 
PHP 6 而 在 网 上 或 PHP.net 站 点 没有 找到 任何 相 天 内 容 的 话 ， 不 必 担 心 ， 
你 并 没有 错过 什么 。 


在 本 书 中 ， 你 将 学 到 配置 和 管理 Apache Web 服 务 器 所 必需 的 概念 、 
PHP 编 程 基础 ， 以 及 使 用 和 管理 MySQL 关 系 型 数据 库 系 统 的 方法 。 本 书 
的 目标 是 提供 理解 如 何 无 颖 地 整合 这 些 技 术 的 基础 ， 并 且 教 授 将 它们 整 
合 到 功能 完备 的 Web 站 点 和 Web 应 用 程序 中 的 实用 知识 。 本 书 应 该 是 迈 
器 高 级 站 点 开 及 的 第 一 步 ， 而 不 是 唯一 的 一 步 。 


本 书 的 目标 读者 


本 书 为 那些 对 基于 Web 的 开发 环境 〈 可 能 是 在 Linux/UNIX 或 
Windows 下 ， 甚 至 是 在 Mac OS X 下 ) 具有 一 般 性 理解 的 人 们 量 身 打造 。 
假设 你 已 经 熟悉 了 上 自己 的 操作 系统 ， 并 且 午 握 了 编译 (在 Linux/UNIX 系 
统 上 ) 或 安装 (在 Windows 和 Mac OS XX 系统 上 ) 软件 的 基本 方法 。 


假设 读者 没有 任何 关于 语言 的 知识 ， 这 样 的 读者 可 先 阅 读 介 绍 使 用 
PHP 编 程 的 草 方 。 人 然而， 如果 你 有 使 用 其 他 编程 语言 的 经 验 ， 例 如 
ASP、JSP、Ruby 或 Pearl， 你 会 发 现 这 些 章节 学 起 来 很 容易 ， 因 为 你 已 经 
蚤 相 庄 如 变量 、 控 制 结 构 、 疯 数 、 对 和 象 等 编程 元 系 。 类 似 的 ， 如 果 你 已 
经 使 用 过 其 他 的 数据 库 ， 例 如 Oracle 或 Microsoft SQL Server， 那 会 为 学 
习 和 MySQL 相 关 的 内 容 呐 定 一 个 泽 实 的 基础 。 


唯一 真正 第 要 的 是 ， 你 能 够 理解 使 用 HTML 创 建 鹏 态 Web 内 容 。 如 
琳 你 只 是 刚刚 开始 Web 开 友 ， 应 该 能 够 使 用 本 书 ， 但 是 ， 你 应 该 和 完 考 虑 
阅读 一 个 HTML 教程。 如 条 你 熟悉 了 创建 基本 的 页 面 ， 那 么 你 会 学 得 
好 。 


本 书 的 组 织 结构 


Ne 
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因为 每 个 部 分 的 内 容 部 构建 在 前 面部 分 的 基础 之 上 。 


第 1 部 分 “基础 知识 ”， 提 供 一 个 安装 的 快速 指导 ， 并 市 领 读者 深入 
解 安 装 和 配置 MySQL、Apache 和 和 PHP 的 过 程 。 在 继续 学 习 之 前 ， 
你 至 少 需 要 完成 这 些 课程 中 的 一 和 种， 要么 快速 安装 ， 要 么 是 更 长 的 
详细 安装 过 程 ， 除 非 你 已 经 成 功 地 安装 了 这 些 软件 。 即 便 你 不 需要 
在 自己 的 环境 中 安装 或 配置 MySQL、Apache 和 PHP， 还 是 应 该 浏览 
一 下 这 些 课程 ， 以 便 理 解 它们 相互 交互 的 基础 。 

第 2 部 分 “PHP 语 言 结 构 ”"， 教 授 PHP 语 言 基 础 ， 包 括 数 组 和 对 象 这 样 
J abidin 那些 示例 将 帮助 你 习惯 编写 代码 ， 将 这 些 示 例 上 传 
到 你 的 服务 器 ， 并 测试 其 运行 结果 。 

第 3 部 分 mayne 介绍 了 中 级 应 用 程序 开 及 的 主题 ， 包 括 使 用 
表单 和 文件 、 限 制 访问 以 及 完成 一 些小 项 目 ， 这 些小 项 目 是 设计 用 
来 介绍 一 个 专门 概念 的 。 

第 4 部 分 “PHP 和 MySQL 整 合 "， 介 绍 使 用 数据 库 的 一 般 方法 ， 例 
如 ， 数 据 库 规范 化 ， 以 及 使 用 PHP 来 连接 并 操作 MySQL。 其 中 包含 
SQL 基础 知识 ， 还 包括 特定 于 MySQL 的 函数 和 其 他 信息 。 

第 5 部 分 “基本 项 目 ?”， 介 绍 如 何人 整合 前 面 已 经 学 习 到 的 所 有 知识 ， 
使 用 PHP 和 MySQL 执 行 一 个 特定 任务 。 这 些 项 目 包 括 地 址 泗 、 一 个 
讨论 论坛 和 一 个 基本 的 网 上 商店 。 这 些 例子 都 是 在 一 个 黑白 环境 下 
构建 的 ， 融 是 说 在 美观 性 上 显得 很 简约 。 这 使 你 可 以 把 精力 集中 在 


| 


程序 设计 和 搭建 结构 所 需 的 逻辑 上 ， 而 不 是 在 显示 美观 上 。 
© 第 6 部 分 “管理 和 优化 >， 介绍 管 理 和 优化 Apache 和 MYVySQL 。 它 还 包 
含 了 有 天 虚拟 主机 以 及 建立 一 个 安全 Web 服 务 器 的 信息 。 


如 果 你 发 现 自己 已 经 熟悉 某 个 主题 ， 可 以 跳 过 并 继续 向 前 学 习 。 然 
而 ， 某 些 地 方 会 引用 前 面 的 章节 中 学 习 过 的 特定 概念 ， 因 此 ， 请 注意 必 
须 浏览 一 下 跳 过 的 章节 ， 以 便 保 证 你 的 开发 环境 和 本 书 一 致 。 


在 每 草 的 末尾 ， 都 有 一 些 问答 题 来 测试 你 对 该 草 内 容 的 筝 握 程 度 。 
附加 的 思考 题 则 提供 了 应 用 该 草 知 识 的 万 外 一 种 方式 ， 并 且 引 导 你 在 下 
一 革 使 用 这 些 刚刚 学 习 到 的 知识 。 
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在 各 和 章 中 出 现 的 程序 清单 中 的 所 有 人 代码， 都 可 以 在 随 书 光 盘 中 找 
到 。 也 可 以 从 作者 的 Web 站 点 http://www.thickbook.com/ 下 载 打 包 的 代 
但 。 


目 己 孙 入 代码 ， 在 打字 、 产 生 氏 误 以 及 执行 叫 人 伤 透 脑 筋 鸭 查找 分 
号 错误 的 任务 等 方面 会 有 些 有 用 的 体验 。 然 而 ， 如 条 你 力 要 略 过 这 些 计 
程 并 且 只 是 把 本 书 的 工作 代码 上 传 到 你 的 站 点 ， 也 没 问 题 。 


本 书 体例 


本 书 使 用 不 同 的 字体 来 表示 代 人 码 和 正文 ， 也 通过 这 种 方法 来 帮 你 识 
别 午 要 的 概 翁 。 在 本 书 中 ， 人 代码、 命令 和 你 所 输入 的 或 者 在 计算 机 屏 送 
上 看 到 的 文本 ， 痢 使 用 等 洁 字 体 。 在 正文 中 定义 新 术语 的 地 方 使 用 和 斜 
体 。 此 外 ， 特 别 的 内 容 版 块 部 市 有 图 标 。 


。“ 提 示 ” 给 出 了 和 当前 话题 相关 的 一 段 有 趣 的 信息 。 
o “你 知道 吗 ? 近 出 建议 ， 或 者 教 给 你 执行 一 项 任务 。 
o “注意 ?敬告 你 海 在 的 缺陷 并 说 明 如 何 避 免 它们 。 


248 QuickStart |=) 导 
安装 和 配置 MySQL 
安 疼 和 配置 Apache 


安装 和 配置 PHP 


1a Z QuickStart" 
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台 安 装 软件 包 XAMPP 的 安装 过 程 。 后 续 的 第 2、3、4 章 分 别 介绍 了 如 何 
从 互联 网 上 获取 并 安装 MySQL 、Apache 和 PHP， 从 而 可 以 确保 软件 版 本 
是 最 新 的 。 另 外 ， 这 几 章 还 展开 说 明了 安装 过 程 中 的 每 一 步 ， 以 及 理解 
这 些 技术 如 何 一 起 工作 的 其 他 重要 信息 。 
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你 只 是 想 要 开始 在 本 地 机 器 上 工作 的 话 ， 本 章 也 是 很 好 的 参考 。 


11 使 用 第 三 方 的 安装 包 


第 三 方 安装 包 是 由 最 初创 建 者 以 外 的 公司 或 组 织 所 提供 的 程序 包 。 
在 本 章 中 ， 我 们 将 学 习 如 何 使 用 XAMPP 安 装 包 来 同时 安装 PHP、 
MySQL 和 Apache， 可 以 在 我 们 将 要 使 用 的 任何 操作 系统 上 
(Linux/UNIX、Windows 或 Mac) 完成 安装 。 


除了 因为 我 目 己 使 用 XAMPP 数 年 了 ， 我 选择 在 本 章 中 使 用 它 的 为 
一 个 原因 是 ， 其 名 称 中 带 有 X。X 表 示 这 是 AMPP (Apache. MySQL. 
PHP 和 Perl〉 的 一 个 跨 平 台 安 装 程 序 (Perl 不 是 本 书 的 主题 ， 因 此 ， 将 其 
当做 额外 的 学 习 内 容 ) 。 还 有 两 种 其 他 很 好 的 Apache、MySQL 和 PHP 安 
疙 程序 包 ， 但 它们 针对 特定 的 操作 系统 。 


e WAMP 一 一 在 Windows 上 安装 Apache、MySQL 和 PHP。 参 见 
http://www.wampserver.com/ for more information. 
e MAMP —— Mac £.28Apache. MySQL#AIPHP. Bl, 


http://www.mamp.info/ for more information. 
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序 包 本 号 涉及 到 一 些 工 作 ， 以 便 确 你 这 些 拉 术 的 最 新 版 本 之 间 没 有 冲 
R; 这 还 必须 经 过 一 个 质量 你 证 的 过 程 。 然 而 ， 这 一 方 采 的 优点 是 ， 妆 
你 使 用 安 冯 程序 包 来 安 疼 这些 技 术 的 时 候 ， 升 级 的 过 程 只 不 过 是 运行 一 
个 新 的 安 痛 程序 ， 它 会 负责 为 你 删除 或 升级 所 有 的 文件 。 


接 下 来 的 3 市 介绍 了 XAMPP 的 基本 安 竣 过程。 你 只 需要 阅读 适用 于 


你 的 操作 系统 的 市 。 然 而 ， 一 定 要 阅读 本 半 1.5 节 ， 因 为 它 适 用 于 所 有 
的 操作 系统 。 


1.2 Linux/UNIX F ff) 24 


尽 党 下 面 介 绍 的 过 程 是 在 Ubuntu Linux 系 统 上 测试 过 的 ， 但 这 些 步 
肾 对 其 他 所 有 Linux 或 商业 UNIX 及 布 的 夭 认 安 痛 都 是 一 梓 的 。 在 编 详 
a 你 可 能 过 到 意外 的 错误 信息 ， aerated 
特定 的 操作 系统 的 文档 。 


如 果 你 使 用 本 书 随 书 光盘 中 包含 的 XAMPP 版 本 ， 请 从 这 里 开始 ， 
以 超级 用 户 局 动 〈 作 为 root 登 录 或 作为 一 个 第 规 系 统 用 户 su 登 录 ) ， 并 
在 文件 系统 下 用 /mnt 参 数 加 载 CD-ROML。 
# mount /dev/cdrom /mnt -t is09660 
现在 ， 你 已 经 访问 了 随 书 光盘 中 的 XAMPP 文 件 ， 或 者 你 已 经 从 
http://www.apachefriends.org/en/xampp-linux.html FÈ J RITRAE, AKE 
BEN 2258 27 PRU T o 


作为 一 名 超级 用 户 ， 将 文件 从 随 书 光盘 的 XAMPP 目 录 《或 者 从 下 
载 位 置 ) 复制 到 /opt 目 录 。 在 /opt 目 录 中 ， 解 压缩 已 经 下 载 的 文件 如 下 。 
# tar xvfz xampp-linux-VERSION-NUMBER.tar.gz -C /opt 


在 编写 本 书 时 ，XAMPP 的 版 本 号 是 1.8.0-beta2， 因 此 ， 其 文件 名 是 xampp-linux-1.8.0- 
beta2.tar.gz。 随 后 的 版 本 将 具有 不 同 的 文件 名 ， 因 此 ， 请 相应 地 修改 命令 。 


这 会 创建 一 个 名 为 /opt/lampp 的 目录 ，XAMPP 安 装 于 其 中 。 要 局 动 
XAMPP， 自 完 ， 将 日 录 蝎 改 到 新 创建 的 目录 。 


# cd lampp 


执行 如 下 的 命令 ， 以 启动 XAMPP (启动 Apache 和 和 MySQL) 。 


# ./lampp start 


将 会 看 到 如 下 的 信息 。 


starting XAMPP for Linux 1.8.0... 

XAMPP: Starting Apache with SSL (and PHPS5)... 
XAMPP: Starting MySQL... 

XAMPP: Starting ProFTPD... 

XAMPP for Linux started, 


SEM AWebhRA az ze GIST, FTA —“ Webi bt as H+ Ba A 
http://localhost/xampp/index.php 。XAMPP 服 务 的 菜单 应 该 会 显示 出 
来 ， 如 图 1-1 所 示 。 


到 > 加 三 和 


English | Destech: Francais | Nederlands | Polski Haliano Norsk / Español (7) Portaguis (Brasil) A AiR 


XAMPP 


APACHE 
FRIENDS 





图 1-1 XAMPP 的 菜单 页 面 


所 有 要 做 的 就 是 这 些 ，XAMPP 已 经 在 你 的 机 器 上 安装 了 Apache、 


PHP 和 MySQL， 并 且 ， 你 可 以 看 到 服务 的 状态 。 在 浏览 
http://localhost/xampp/index.php 的 时 候 ， 你 可 以 通过 左边 列 出 的 链接 了 
解 更 多 的 相关 信息 。 


要 俘 止 XAMPP 及 其 服务 ， 可 以 在 任何 时 候 通 过 命令 行 执 行 如 下 命 


+. 
# fapt/lampp/lampp stop 


确保 阅读 本 章 1.5 节 ， 了 解 有 关 保 护 安 装 了 XAMPP 的 机 需 的 更 多 信 
电 〈 即 使 该 机 器 只 是 用 于 开发 ) 。 


1.3 7£ Windows | 224 XAMPP 


随 书 光盘 中 的 XAMPP 安 六 文件 已 经 经 过 了 测试 ， 并 且 适 用 于 从 
Windows XP 到 Windows 7 的 多 种 Windows 操 作 系 统 。Windows 较 早 的 版 
本 则 不 文 持 。 要 使 用 随 书 光盘 中 的 XAMPP 安 痛 文 件 ， 首 乞 将 光盘 插入 
光驱 ， 它 应 该 会 日 动 播 放 。 如 果 没 有 ， 双 击 “ 我 的 电脑 * 下 的 光驱 图 标 ， 
并 且 找 到 包含 XAMPP 安 装 文 件 的 目录 。 


找到 随 书 光盘 上 的 XAMPP 文 件 ， 或 者 从 
http://www.apachefriends.org/en/xampp- windows.html FE J RITRAE Z 
fa, IUTE A IE A E D r E TENY o 


H + Windows#e E 22 22 7H MAS AY ME, H h FWindows#L 
as LAY fe 2488 S ARRERA ES. UREN eR RS A I BE 
ÍT. WW aD Fhttp://www.apachefriends.org/en/faq-xampp-windows.html 
的 针对 Windows 用 户 的 XAMPP FAQ. 


首先 会 请 你 选择 语言 ，English 是 默认 的 选择 。 在 选择 了 语言 并 单 击 
OK 按钮 之 后 ， 你 将 会 看 到 安装 程序 的 欢迎 界面 ， 如 图 1-2 所 示 。 


Welcome to the XAMPP 1.7.7 Setup 
Wizard 


This wizard will guide you through the installation of XAMPP 
L 7.7. 


Itis recommended that you dose all other applications 
before starting Setup. This will make it possible to update 
relevant system files without having to reboot your 
computer. 


Click Next to continue, 





图 1-2 XAMPP 安 装 主 界面 


au a 
VE: 


在 编写 本 书 的 时 候 ，XAMPP 的 版 本 号 是 1.7.7， 因 此 ， 安 装 向 导 显 示 了 该 版 本 号 。 最 新 的 
版 本 会 使 用 不 同 的 版 本 号 ， 但 是 ， 安 装 过 程 是 类 似 的 。 

点 击 Next 按 钮 以 继续 安装 过 程 。 就 像 大 多 数 基于 向 导 的 安装 一 样 ， 
在 进行 下 一 步 之 前 ， 安 装 程序 要 求 你 选择 一 个 安装 位 置 和 一 些 安装 先 
项 。XAMPP 的 安装 也 没什么 不 同 ， 你 应 该 保留 默认 的 安装 位 置 ， 以 及 
默认 的 安装 选项 ， 并 且 点 击 Next 按 钮 继续 进入 下 一 个 界面 。 此 时 ， 安 装 
过 程 自动 进行 ， 如 图 1-3 所 示 。 


Installing 
Please wait while XAMPP 1.7.7 is being installed, 


Extract: wardze? OO 


unknown. gif 
unknown. png 

: up.gif 

: Up.png 

: uu. gif 
uu.png 

: uuencoded. gif 

: uuencoded.png 
world Laif 
world 1.png 
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图 1-3 XAMPP 安 装 继续 ， 解 压 文件 


安装 过 程 完成 之 后 ， 安 装 程序 会 提示 你 当前 状态 ; 蛙 击 Finish 按 钮 
完成 安装 。 在 XAMPP 安 装 过 程 结束 之 击 ， 它 询问 你 是 否 想 要 局 动 
Control Panel， 以 管理 所 安装 的 服务 ， 如 图 1-4 所 示 。 


[E] XAMPP 1.7.7 win32 
installation Complete 
Setup was completed successfully. 


|| E XAMPP 1.7.7 win32 


Congratulations! The installation was successful! Start the XAMPP Control Panel 
now? 
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图 1-4 XAMPP 238 52 BX 


XAMPP Control Panel 使 你 能 够 通过 一 次 鼠标 单 击 来 局 动 并 俘 止 在 
机 器 上 运行 的 Apache 和 MySQL 服 务 器 进程 ， 如 图 1-5 所 示 。 如 果 你 只 是 
为 了 进行 开发 而 在 本 地 机 右上 运行 这 些 服务 占 进 程 ， 只 有 当 你 需要 它们 
的 时 候 ， 才 想 要 打开 它们 ; Control Panel 允 许 你 快速 地 做 到 这 一 点 。 


加] XAMPP Control Panel v3.0.11 


Modules 
Service Module PID(s) Port(s) Actions 


Apache 5436 80 [ Stop | 


“Admin | 


FileZilla 
Mercury Admin 


Admin | 


Tomcat Start 


v [apache] Uninstall the service manually first 
[mysql] MySQL Service Detected With Wrong Path 
vi [mysql] Uninstall the service manually first 
vi [mysql] Possible problem detected! 
vi [mysql] Port 3306 in use by "mysald.exe"! 
[main] Starting Check-Timer 
vi [main] Control Panel Ready 
Vv [apache] otarting apache service... 
| [apache] status change detected: running 

















图 1-5 XAMPP Control Pane 


EM iWebAhs a8 2 mM, FTF Webb it as Ft ALLA 
http:/localhost/xampp/xampp.php 。 应 该 会 显示 XAMPP 束 单 ， 如 图 1-6 
所 示 。 


j Exames 


€ C © localhost/xampp/index.php 


=) XA M P P f r Wi 门 W English Deutsch / Francais / Nederlands / 
Polski) Italiano / Norwegian į Español; =x; 
Zi 


Portugués (Brasil); 号 


XAMPP 
22) Welcome to XAMPP for Windows! 


Congratulations: 
You have successfully installed XAMPP on this system! 


Now you can start using Apache and Co. You should first try »Status« on the left navigation to 
eevee make sure everything works fine, 
Components : 
| For OpenSsl support please use the test certificate with httos://127.0.0.1 or httos://localhost 
H ees Ri | Good luck, Kay Vogelgesang + Kai ‘Oswald’ Seidler 
CD Colecon 
Biorhythm | 
Instant Art- 
Phone Book | 


oo Pn 
perlinfo() Fe 
Guest Book : 


AEE 
| Status | 
[Tomcat eramplesy.! 


phpMyAdmin | 

Webalizer | 
Mercury Mail: 
FileZilla FTP 





图 1-6 XAMPP3é 42.00 [A 


所 有 要 做 的 就 是 这 些 ，XAMPP 已 经 在 你 的 机 器 上 安装 了 Apache、 
PHP 和 MySQL， 并 且 ， 你 可 以 看 到 服务 的 状态 。 在 浏 贤 
http://localhost/xampp 的 时 候 ， 你 可 以 通过 左边 列 出 的 链接 了 解 更 多 的 
相关 信息 。 确 保 阅 读本 章 1.5 站 ， 了 解 有 关 保 护 安 装 XAMPP 的 机 右 的 更 
多 信息 (即使 该 机 颖 只 是 用 于 开 友 ) 。 


1.4 在 MacOS X 上 安装 XAMPP 


随 书 光盘 中 的 XAMPP 安 装 文件 已 经 经 过 了 测试 ， 并 且 适 用 于 Mac 
OSX 1.4 及 其 之 后 的 各 个 版 本 。Mac OS X 的 较 早 的 版 本 则 不 支持 。 


要 使 用 随 书 光 盘 中 的 XAMPP 安 痛 文 件 ， 首 和 匈 将 光盘 插入 Mac 上 的 
光驱 ， 它 应 该 会 日 动 播放 。 如 末 没 有 ， 双 击 “ 我 的 电脑 ”下 的 区 驱 图 标 ， 
并 且 找 到 包含 XAMPP 安 装 文件 的 目录 。 


找到 随 书 光盘 上 的 XAMPP 文 件 ， 或 者 从 
http://www.apachefriends.org/en/xampp-mac.html FE J ITRE ZJE, 
Wi AIC Wa Sir ERRET. MARE RU 1-7 BAN A FR 
KE 


-| XAMPP for Mac OS X 1.7.3 
2 items, 318.65 GB available 


=] XAMPP for Mac OS X 


Version 1.7.3 


RE a E T 
MERT., Eea T 
E AI. rE 

Porat TEELT IETT a a e, 
A ey a UA LEY in TAHI 


Applications 





图 1-7 在 Mac 上 安装 XAMPP 的 说 明 


在 编写 本 书 时 ， 针 对 Mac 用 户 的 XAMPP 的 版 本 号 是 1.7.3， 因 此 ， 
这 个 安装 程序 包 显 示 了 该 版 本 号 。 随 后 的 版 本 将 具有 不 同 的 文件 名 ， 
此 ， 请 相应 地 修改 命令 。 


按照 界面 上 的 指示 ， 将 XAMPP 文 件 夹 皂 忠 到 Applications 文 件 夹 
中 。 在 复制 了 文件 炎 和 文件 之 后 ， 可 以 在 /Applications/XAMPP 文 件 夹 中 
找到 XAMPP Control Panel 的 链接 ， 如 图 1-8 所 示 。 


6 items, 94.17 GB available 
Poe oe ALLE ea aii ies Be aR RRS ee pt P Date Modified Kind 
al cgi-bin Feb 27, 2010 2:00 PM Alias 
Feb 27, 2010 2:00 PM Alias 
Feb 27, 2010 2:00 PM Alias 


Feb 27, 2010 2:00 PM Alias 


XAMPP Control Feb 27, 2010 2:00 PM =- Application 
4 (aad xamppfiles Feb 27, 2010 2:01 PM -- Folder 





图 1-8 找到 XAMPP Control Panel 的 链接 


双击 该 链接 以 打开 XAMPP Control Panel， 通 过 该 面板 我 们 可 以 在 
机 器 上 局 动 或 停止 Apache 和 和 MySQL 服务 器 进程 。 如 果 你 只 是 为 了 进行 
开 友 而 在 本 地 机 絮 上 运行 这 些 服务 右 进 程 ， 只 天 当 你 需要 它们 的 时 候 ， 
才 想 要 打开 它们 ; Control Panel 人 允许 你 快速 地 做 到 这 一 点 。 


要 测试 Web 服 务 絮 是 否 运 行 ， 打 开 Web 浏 贤 龙 并 且 输 入 
http://localhost/xampp/index.php 。XAMPP 服 务 的 荣 单 应 该 会 显示 出 
来 > 加 图 1-9 所 示 O 


XAMPP for Mac OS X 1.7.3 


English / Deutsch / Francais / Nederlands / Polski / Italiano / Norsk / Español / 中 文 / Portugués (Brasil) / BAR 


Welcome to XAMPP for Mac OS X 1.7.3! 


Congratulations: 
You successfully installed XAMPP on this system! 


Now you can start using Apache and Co. Firstly you should try »Status« on the left navigation to make sure everything works fine. 
After testing you may take a look at the examples below the test link. 


If you want to start programming PHP or Perl (or whatever ;) please take a look at the XAMPP manual first and get more information about your XAMPP installation. 
Good luck, 
Kristian Marcroft, Florian Polini, Christian Speich & Team 





图 1-9 XAMPP 菜 单 页 面 


所 有 要 做 的 就 是 这 些 ，XAMPP 已 经 在 你 的 机 器 上 安装 了 Apache、 
PHP 和 MySQL， 并 且 ， 你 可 以 看 到 服务 的 状态 。 在 浏 贤 
http://localhost/xampp/index.php 的 时 候 ， 你 可 以 通过 左边 列 出 的 链接 
TEREKE. MARREK ELS, ARAI XAMPP 
MHLE Baa CRU AALA REFF) 。 


1.5 ikKXAMPPH ZA 


XAMPP 的 主要 目的 是 提供 一 种 快速 容易 的 方式 ， 以 便 在 开 友 坏 培 
中 安装 Apache、MySQL 和 PHP。 这 种 快速 而 容易 的 安装 的 代价 之 一 ， 奈 
定安 全 设置 不 完整 ， 或 者 说 ， 至 少将 确定 安全 设置 是 售 足 人 够 重要 的 权利 
交 给 了 了 用户。 


使 用 最 新 的 安 竣 ， 会 有 如 下 的 一 些 潜在 的 安全 问题 。 


。 MySQL 管理 员 用 户 没 有 密码 (你 可 以 使 用 一 个 空白 的 密码 )。 

e 以 同样 的 用 户 运 行 MySQL 和 Apache (nobody， 这 只 针对 
Linux/UNIX 和 Mac 操 作 系 统 ) 。 

。 一 些 服务 是 可 以 通过 该 网 络 访问 的 ， 除 非 你 通过 目 己 的 防火 场 明 确 
地 禁止 访问 。 


然而 ，XAMPP 为 每 个 操作 系统 都 提供 了 一 种 工具 ， 通 过 它 ， 我 们 
可 以 在 一 种 开发 环境 中 运行 和 完成 增强 XAMPP 系 统 安 全 性 的 过 程 。 
。 在 Linux/UNIX 上 ， 通 过 输入 如 下 命令 来 运行 该 工具 。 
# /opt/lampp/lampp security 
e {Windows E, ii fe Webixl hi as P Y7 IA) 
http://localhost/xampp/index.php , J-M Æ (AY SRALSe p 28 
Security, J m tJ FF Security Console. 
。 ŒMacE, HIMA TME RIASA i a O 


# /Applications/XAMPP/xamppfiles/xampp security 


1.6 ”故障 排除 


本 章 的 所 有 步 又 已 经 用 本 书 附带 的 光盘 里 包含 的 软件 版 本 测试 过 
如 果 遇 到 安装 问题 ， 首 先 检查 是 否 严格 按照 本 章 给 出 的 步 又 进行 


然后 ， 查 看 人 的 XAMPP Web 站 点 ， 
以 了 解 适用 于 这 一 安 少 包 的 FAQ。 


如 采 这 些 过 程 仍 然 不 管用 ， 并 且 你 想 要 答 试 其 shinee hws 
4, MA, A 清 自行 尝 SRA (在 本 章 开 始 处 提 到 ) 。 你 也 
以 笑 试 使 用 “ 较 复 林 ” 的 安装 方式 ， 这 需要 使 用 后 续 3 章 中 介绍 的 扩 A 
o IXH ae ERARE. DI HT LIFE ERE R 2 ae |] E HS 
其 他 站 点 的 链接 。 


第 2 章 ” 安 装 和 配置 MySQL 


在 本 章 中 ， 你 将 学 到 : 


。 如 何 安 站 MySQL。 
。 运行 MySQL 的 基本 安全 规则 。 
。 如 何 通 过 用 户 授 权 系 统 来 使 用 MySQL 。 


这 是 与 复杂 安装 相关 的 三 章 中 的 第 一 章 ， 在 此 你 将 学 习 如 何 安 装 开 
发 环境 。 如 果 你 打算 使 用 MySQL 和 PHP， 我 们 首先 要 介绍 MySQL 的 安 
装 ， 因 为 在 某 些 系统 中 编译 PHP 模 块 需要 完成 MySQL 的 安装 才 行 。 


2.1 MySQL i) = Hy We AS A ARR A AS 


本 章 的 安装 说 明 针对 的 是 MySQL Community Server 5.5.21， 这 是 
MySQL 软 件 的 当前 产品 版 本 。 这 个 版 本 号 可 以 读 作 “MySQL 服 务 器 软件 
的 主 版 本 5， 次 版 本 《小 发 布 ) 5 的 第 21 次 修订 ”。 修 订 版 和 小 的 发 布 并 
不 遵从 既定 的 一 系列 发 布 计 划 。 当 对 代码 进行 扩展 和 修复 并 且 进 行 了 彻 
底 的 测试 后 ，MySQL AB 吏 会 用 一 个 新 的 修订 号 或 次 版 本 号 来 及 布 一 个 
新 的 版 本 。 


当 你 购买 本 书 的 时 候 ， 可 能 次 版 本 号 已 经 变 成 了 5.5.22 或 者 更 辟 。 
如 果 是 这 种 情况 ， 你 可 以 阅读 位 于 
http://dev.mysql.com/doc/refman/5.5/en/news-5-5-x.html 的 列表 来 了 解 安 
装 或 配置 过 程 中 的 任何 变化 ， 那 里 的 内 容 构 成 了 本 章 的 大 部 分 内 容 。 


尺 官 在 两 个 次 版 本 更 新 之 间 不 可 能 所 有 的 安 北 过 程 部 要 变化 ， 但 你 
还 是 应 该 养 成 习惯 查看 自己 所 安装 和 维护 的 软件 的 更 新 日 志 。 如 果 在 你 
六 读本 书 的 时 候 ， 确 实 出 现 了 一 个 钦 版 本 的 变化 ， 但 更 新 日 志 中 并 没有 
提 人 到 和 安 北 的 变化 ， 你 只 需要 用 心 记 下 ， 并 且 当 出 现在 本 书 的 安 站 说 明和 
相应 的 图 时 ， 用 新 的 版 本 写 蔡 代 束 行 了 。 


2.2 ”如 何 获 取 MySQL 


MySQL AB 走 负责 开 及 、 维 护 和 及 布 MySQL 数 据 库 的 公司 的 名 字 ， 
经 过 一 系列 的 收购 之 后 (Sun Microsystems AI% MySQL AB，Oracle 公 
司 收购 了 Sun Microsystems) ， 现 在 ， 数 据 库 巨 人 Oracle 拥 有 MySQL 。 
然而 ， 该 软件 的 MySQL Community Edition 版 本 一 直 保 持 开 源 ， 它 是 由 
开源 开 及 者 文 持 的 ， 并 且 可 以 在 MySQL 的 Web 站 扣 
http://www.mysql.com/ 免费 获取 。 上 所 有 平台 的 二 进 制 发 布 ， 用 于 Mac OS 
X 的 安装 程序 包 ， 以 及 用 于 LinuxUNIX 平 台 的 RPM 和 源 代码 文件 ， 都 可 
以 获得 


提示 : 





Linux 和 和 Mac OS X 发 布 通常 会 包 售 MySQL 软 件 的 某 个 版 本 或 其 他 开源 的 MySQL 软 件 ， 尽 
管 在 当前 的 发 布 版 本 之 后 通常 je 几 个 修订 版 或 者 小 版 本 。 


本 章 的 安 竣 说 明基 于 正式 及 布 的 MySQL 5.5.x Community Serverhix 
本 。 所 有 的 文件 都 可 以 从 http://dev.mysql.com/downloads/mysql/5.5.html 
下 载 到 ， 并 且 在 编号 本 书 的 时 候 所 采用 的 当前 版 本 也 可 以 在 下 载 资 料 中 
找到 。 


2.3 ”在 Linux/UNIX 上 安装 MySQL 


不 管 你 是 使 用 RPM 还 是 二 进 制 代 人 码 安 帮 ， 在 LinuxUNIX 上 安 竣 
MySQL 都 比较 简单 。 如 果 你 通过 RPM 安装 ，MySQL AB 提 供 了 专门 平 
人 台 的 RPM， 例 如 针对 运行 在 不 同 尖 型 的 处 理 颖 (如 32 位 或 64 位 的 x86) 
上 的 SuSE Linux 或 一 般 有 的 Linux。 


对 于 RPM 的 一 个 最 小 的 安 半 ， 你 需要 如 下 来 目下 载 页 
Mhttp://dev.mysql.com/downloads/mysql/5.5.html 的 两 个 文件 (或 者 从 本 
书 随 书 光 盘 获 取 的 文件 ) 。 


e MySQL-server-type - VERSIONNUMBER .PLATFORM .rpm — The 
MySQL server 

e MySQL-client-type - VERSIONNUMBER .PLATFORM .rpm — The 
standard MySQL client libraries 


BEAT I/D) IRPM 2228, H BTA MH FATS 


# rpm -i My$QL-server-VERSION.1386.rpm My$QL-client-VERSION.1386. rpm 


提示 : 





把 文件 名 中 的 VERSIONNUMBER 蔡 换 为 你 所 下 载 的 实际 版 本 ， 并 且 把 PLATFORM 蔡 换 
为 你 所 使 用 的 平台 的 缩写 。 例 如 ， 当 前 的 针对 通用 的 Linux 发 布 的 MYSQL 5.0 Server RPM, MY 
做 MySQL-server- standard-5.0.51a-0.sles10. i586.rpm， 而 客户 疹 库 RPM 叫做 MySQL-client- 
standard-5.0.51a-0.sles10.i586. rpm. 


对 于 Debian 包 的 安装 ， 你 需要 位 于 


http://dev.mysql.com/downloads/mysql/5.5.html 的 页 面 中 的 *.deb 文 件 〈 或 
者 从 本 书 随 书 光 枪 中 获取 〉。 然 后 ， 在 命令 提示 行 中 输入 如 下 命令 。 
# dpkg -i mysql-VERSION-debian6.@-1686.deb 


ARRP HIR is OL RK EI I PS a Hl (AS A ER 
安装 。 这 个 方法 需要 gunzip 和 tar 工 具 解 压缩 并 拆 包 有 发布， 并且 还 需要 能 
够 在 系统 上 创建 组 和 用 户 。 二 进 制 代码 发 布 安 冯 过 程 中 的 第 一 组 命令 ， 
MERI TAA TPH Ra ARH, IRA 


# groupadd mysql 
# useradd -r -g mysql mysql 


# cd /fusr/local 
# tar zxvf /path/to/mysql -VERSION -PLATFORM. tar.gz 


提示 : 


把 文件 名 中 的 VERSION-PLATFORM 蔡 换 为 你 实际 所 下 载 的 版 本 。 例 如 ， 当 前 的 MySQL 
5.5 通 用 Linux 二 进 制 代 码 发 布 叫做 mysql-5.5.21-1- linux2.6.i386.tar。 


接 看 ， 指 令 使 用 一 个 较 短 的 名 字 创 建 一 个 从 与 链接 。 


# In -s mysql-VERSION-PLATFORM mysql 
# cd mysql 


一 旦 拆 包 ，README 文 件 和 INSTAILEL 文 件 将 会 根据 你 所 选择 的 
MySQL 版 本 来 完成 剩 下 的 安装 过 程 。 通 常会 用 到 如 下 的 一 组 命令 。 


scripts/mysql_ install db --user=mysql 
chown -R root 

chown -R mysql mysql data 

chgrp -R mysql . 

bin/mysqld safe --user=mysql & 


EEH H 


RIIE CAERA MySQL a, Hr AFET ERRE 


PAUP RF HI OAS ESA. WORE EP A BE AY ee, a 
i] 2.6 ZR HL ERR”? — T o 


2.4 ”在 Mac OS X 上 安装 MySQL 


Mac OS X 下 的 MySQL 安 痛 过 程 相当 人 简单，MySQL AB 的 开 友 者 已 
经 为 Mac OS X 创 建 了 一 个 安装 包 。 到 位 于 
http://dev.mysql.com/downloads/mysql/5.5.html 的 MySQL 下 载 页 面 并 且 找 
到 Mac OS X《〈 或 者 使 用 随 书 光盘 中 的 文件 ) 。 如 果 你 访问 该 站 点 ， 确 
体 下 载 了 适合 你 的 系统 的 DMG: 无 论 你 使 用 的 是 Mac OS X 10.5 或 10.6 
版 ， 或 者 32 位 或 64 位 的 系统 。 当 你 有 有 了 该 文件 ， 双 击 DMG 文 件 夹 。 在 
打开 了 了 DMG 存档 之 后 ， 你 会 看 到 一 个 文件 夹 其 中 有 有 一些 文 件 ， 如 图 2-1 
所 示 。 


,mysql-5.5.21-05x10.6-x86_64 


4 items, 3 MB available 


v 


mysgl-5.5.21-0sx10.6- MySQLStartupltem.pkg ReadMe. txt 
“X86 64.pkq 





图 2-1 显 示 MySQL DMG 文 件 夹 的 内 容 


当下 载 结束 ， 拆 包 安 装 文件 并 且 双 击 *.pkg 文 件 。 后 续 的 安装 步骤 
完成 如 下 过 程 。 


1. MySQL 安 痛 包 目 动 司 动 〈 如 网 2-2 所 示 ) > Mi Continueik F< F 
下 一 步 。 


@ Install MySQL 5.5.21-community for Mac OS X 


Welcome to the MySQL 5.5.21-community for Mac OS X Installer 


a N 
© Introduction ©. » 


You will be guided through the steps necessary to 
install this software. 


@® Read Me 


i 
| Continue | 
sos | 





图 2-2” Mac 的 MySQL 安 装 程 序 已 经 启动 


2. 接 下 来 的 几 个 界面 将 会 包含 与 安装 A 
通用 信息 。 疯 谈 这 些 界 面 的 内 容 并 单 击 Continue 按 钮 通过 它们 。 


.在 通过 了 信息 和 授权 许可 界面 之 后 ， 必 须 选择 一 个 安装 目标 。 
oni 然后 单 击 Continue 按 钮 。 


__ ‘= Install MySQL 5.5.21-community for Mac OS X 
Select a Destination 


E Select the disk where you want to install the MySQL 
Ə Introduction è, X software. 


© Read Me 


CON 
Ə License ,局 — 


-TEN ai 


© Destination Select 4 


Mothership 
95.1 GB free 
319.73 GB total 


( GoBack | { Continue ) 


Seine = o S 





图 2-3 ”选择 一 个 安装 位 置 


A, 下 一 步 会 验证 安装 位 置 选 择 ， 并 且 和 需要 蛙 击 Install 按 钮 来 继续 。 
此 时 ， 在 这 个 安装 过 程 继续 进行 之 前 ， 可 能 提示 你 输入 管理 员 用 户 名 和 
蜜 但 。 一 旦 继续 开始 ， 束 让 此 过 程 运 行 ， 和 直到 安 半 完成 ， 如 图 2-4 上 所 
示 。 


æ Install MySQL 5.5.21-community for Mac OS X 
The installation was completed successfully. 


‘i, ` LN 
© Introduction ©), 


| 
© Read Me Va, | 
© License ,6 D Wi 


© Destination Select ™ 


e installation Tye | The installation was successful. 


© Installation ， 


@ Summary | 


The software was installed. 








图 2-4 MySQL 已 经 安装 


安装 包 中 还 包含 了 MVySQL Startup Item 安 装 程序 。 如 果 希 望 MySQL 
在 系统 局 动 的 时 候 自 动 局 动 ， 束 安装 这 个 附加 的 包 。MySQL Startup 
Item 的 安装 开 从 刚才 所 搬 述 的 标准 安装 方法 ， 即 双击 *.pkg 文 件 ， 选 择 一 
个 目标 盘 ， 然 后 让 安装 过 程 运行 到 完成 。 安 家 MySQL Startup Item 之 
后 ， 在 命令 行 窗口 中 使 用 如 下 的 命令 来 局 动 MySQL。 


# sudo /Library/StartupItems/MySQLCOM/MySQLCOM start 


当 你 试 匈 局 动 MySQL 的 时 候 ， 可 能 要 求 你 输入 管理 员 密 但 。 在 局 
动 了 了 MySQL 之 后 ， 按 下 CtrltD 退 出 命令 行 窗口 。 启 动 了 了 MySQL 之 后 ， 
我 们 可 以 跳 到 2.7“ 基 本 安全 规则 ”一 节 。 如 果 在 安装 中 碰 到 任何 问题 ， 请 
查阅 2.6“ 安 装 故 障 排 除 ” 一 节 。 


2.5 在 Windows 上 安装 MySQL 


Windows 上 的 MySQL 安 装 过 程 使 用 一 个 标准 的 Microsoft Windows 安 
装 程序 (Microsoft Windows Installer, MSI) 文件 来 完成 在 Windows 
XP. Windows Server 2003、Windows Vista 或 Windows 7#L4¢ _KMySQL 
的 安装 和 配置 过 程 。 到 位 于 
http://dev.mysql.com/downloads/mysql/5.0.html 的 MySQL 下 载 页 面 ， 并 且 
找到 标题 为 “Windows Downloads” 的 小 节 。 下 载 Windows Essentials X 
件 ， 其 扩展 名 为 *.msi。 下 载 完 这 个 文件 ， 双 击 文件 开始 安 猴 过 程 。 


如 下 的 步骤 给 出 了 使 用 MySQL AB 的 Windows Essentials 28 Fe FF Œ 
Windows 上 安 痛 MySQL 5.0.20 的 细 贡 。 不 管 你 的 Windows 环 境 是 什么 ， 
安装 过 程 将 办 从 同样 的 步 双 C。 


yy. ae 
VERS: 





Windows 用 户 也 可 以 使 用 ZIP Archive 版 本 。 如 果 你 想 要 安装 ZIP Archive 版 本 ， 确 保 阅读 位 
于 http://dev.mysql.com/doc/refman/5.5/en/windows-choosing-package.html 的 MySQL 手 册 的 说 明 
和 介绍 o 


BRA RR, RU PAPIRET o 


1. 双击 *.msi 文 件 开始 安装 过 程 。 你 将 会 看 到 安装 问 导 的 第 一 个 界 
面 ， 如 图 2-5 所 示 。 单 击 Next 按 钮 继续 。 


期 | MySQL Server 5.5 Setup 


Welcome to the MySQL Server 5.5 Setup 
Wizard 


The Setup Wizard will install MySQL Server 5.5 on your 
computer, Click Next to continue or Cancel to exit the Setup 
Wizard, 


Copyright (ci 2000, 2011, Orade and/or its affiliates, All 
rights reserved, 





图 2-5 Windows f MySQL 2228 [A] SA 38 — 2 


2. WEZENN. Typical (典型 ) . Complete (EE) 或 
Custom (HX) 〈 如 图 2-6 所 示 ) 。 RR 
痰 的 MySQL 组 件 ， 而 Complete 选 项 则 会 安 状 MySQL 的 所 有 组 件 ， 包 括 
MMT AE. Typical ze seein TRE RUA, ANE ete 
T FART MySQLi#t {7 — ATE ee BRA OL. ARS REFURB. OTE 
Typical ZRIN. FFA Bc Next#Z 4H Ak 2 . 


je) MySQL Server 5.5 Setup 


Choose Setup Type 
Choose the setup type that best suits your needs 


Installs the most common program features, Recommended for most users, 


Allows users to choose which program features will be installed and where 
they will be installed. Recommended for advanced users, 


All program features will be installed, Requires the most disk space, 





图 2-6 ”选择 一 种 安装 方式 


3. 确认 在 下 一 个 界面 中 的 选择 并 且 单 击 Immstall 按 钮 继续 。 安 装 过 程 
负 贡 把 文件 安装 到 正确 的 位 置 。 


当 安 北 过 程 完 成 后 ， 可 以 选择 继续 MySQL Configuration 


Wizard (MySQL A I) 。 强 烈 推 荐 运行 这 个 同 导 ， 因 为 它 会 创建 
一 个 目 定 义 的 my.ini 文 件 ， 它 根据 你 的 具体 需求 而 设置 。 要 继续 进行 
MySQL A P, XF Configure the MySQL Server Now# it, JfH. 
单 击 Finish 按 钮 ， 如 图 2-7 所 示 。 


| fe MySQL Server 5.5 Setup 


Completed the MySOL Server 5.5 Setup 
Wizard 


Click the Finish button to exit the Setup Wizard. 





图 2-7 ”现在 ，MySQL 已 经 安装 了 ， 继 续 执 行 MYSQL Config uration Wizard 


5. 你 将 看 到 配置 癌 导 的 欢迎 界面 ， 单 击 Next 投 钮 来 继续 同 导 的 下 
一 个 步骤 。 你 将 会 看 到 服务 磊 配 置 的 两 个 选项 : Detailed 和 Standard。 我 
们 使 用 Detailed Configuration 先 项， 这样 可 看 到 可 用 的 所 有 选项 。 如 末 
决定 选择 Standard Configuration 选 项 ， 必 须 手 动 修 改 文件 my.ini 以 达到 想 
要 的 配置 。 选 择 Detailed Configuration 单 选 按钮 ， 然 后 单 击 Next 按 钮 继 


sb 
Z5 o 


6. 必须 在 如 网 2-8 所 示 的 弄 面 中 做 出 下 一 个 选择 。 在 这 个 步骤 中 ， 
你 要 选择 所 运行 的 机 器 类 型 : Developer Machine, Server Machine 或 者 
Dedicated MySQL Server Machine。 在 这 个 界面 上 的 选择 决定 了 所 用 的 内 
存 、 便 盘 和 处 理 右 的 分 配 。 如 末 你 为 了 测试 而 在 个 人 电脑 上 使 用 


一 一 人 


MySQL， 则 选择 Developer Machine 选 项 。 如 果 MySQL 所 运行 的 机 右上 


还 有 其 他 的 服务 器 软件 ， 并 且 比 你 在 个 人 电脑 上 运行 MySQL 要 占用 更 
多 的 系统 资源 ， 那 就 选择 Server Machine 选 项 。 如 果 MySQL 是 机 器 上 所 
运行 的 主要 服务 ， 并 且 可 以 使 用 大 量 的 系统 资源 ， 选 择 Dedicated 
MySQL Server Machine 选 项 。 在 做 出 选择 之 后 ， 单 击 Next 按 钮 继续 。 


MySQL Server Instance Configuration Wizard 


MySOL Server Instance Configuration 


Configure the MySQL Server 5.5 server instance, 


Please select a server type. This will influence memory, disk and CPU usage, 


memory. 


C Server Machine 


Several server applications will be running on this machine. 
a Choose this option for web/application servers. MySQL will have 
=< medium memory usage, 
C Dedicated MySQL Server Machine 


P This machine is dedicated to run the MySQL Database Server. No 
T other servers, such as a web or mail server, will be run. MySQL 
i will utilize up to all available memory. 


CO caret | 





图 2-8 选择 服务 需 交 型 作为 MySQL 配 置 的 一 部 分 


7. 下 一 个 配置 选项 适合 于 数据 库 应 用 。 这 些 选 项 是 Multifunctional 
Database. Transactional Database 和 Non-Transactional Database Only. Xf 
Multifunctional Database，InnoDB 和 MyISAM 存 储 引 擎 二 者 之 间 可 以 
平均 地 分 配 资 源 。Transactional Database 也 支持 ImhnoDB 和 MyISAM， 但 
FE K & BRS 4s ot V5 (|=) FInnoDB. Non-Transactional Database Only 则 
不 文 持 InnoDB， 并 且 把 所 有 的 资源 应 用 于 MyISAM。 除 非 你 确切 地 知道 
目 己 的 数据 库 使 用 哪 种 存储 引擎 ， 人 否则 选择 Multifunctional Database!#. i 


FEIE h Nextt tH ARE 


8. 如 条 已 经 选择 了 一 个 数据 库 使 用 选项 ， 其 中 包含 了 InnoDB 存 储 
引擎 ， 配 置 过 程 的 下 一 步 就 允许 配置 硬盘 位 置 和 存储 闵 值 。 默 认 的 情况 
如 图 2-9 所 示 ， 可 以 单 击 Next 按 钮 继续 简单 地 确认 默认 配置 ， 或 者 可 以 
修改 这 些 设 置 ， 然 后 单 击 Next 按 钮 继续 ， 从 而 让 自 定 义 设 置 起 作用 。 


MySQL Server Instance Configuration Wizard 


MySOL Server Instance Configuration 
Configure the MySQL Serer 5.5 server instance, 
Please select the drive for the InnoDB datafile, if you do not want to use the default 
settings. 
InnoDB Tablespace Settings 


(<j Please choose the drive and directory where the InnoDB 
ap tablespace should be placed. 


ce =| [Installation Path -| =] 


Drive Info 


Volume Name: 
File System: 


| 


国 60 GB Diskspace Used [|_| 521.4 GB Free Diskspace 


i Cancel | 





图 2-9 AyInnoDB 4¥ figi 5| SE yd A A AE FD ae 


9. 下 一 个 配置 选项 决定 了 了 MySQL 服务 左 的 并 及 连接 数 。 你 的 设置 
取决 于 Web 站 点 或 应 用 程序 所 使 用 的 数据 流 和 数据 库 的 数量 。 上 默认 的 设 
置 是 Decision Support (DSSYOLAP， 最 多 100 个 并 发 连接 ， 假 定 平均 值 为 
20 个 。Online Transaction Processing (OLTP) 选 项 的 最 多 并 发 连接 数 是 500 
个 ， 而 Manual 设 置 允 许 我 们 从 下 拉 列 表 中 选择 一 个 数值 或 者 目 己 键入 一 
个 数值 。 做 出 自己 的 选择 并 单 击 Next 按 钮 继续 。 


10. 配置 过 程 中 的 下 一 步 是 Networking Options 界 面 。 在 这 里 ， 我 们 
可 以 激活 或 者 关闭 TCP/PP 了 网络， 并 且 可 以 配置 连接 到 MySQL 的 冰 口 
写 ， 默 认 是 3306， 实 际 上 我 们 可 以 使 用 任何 没有 使 用 的 端口 。 这 个 界面 
中 的 另 一 个 选项 可 以 打开 或 关闭 严格 模式 ， 推 荐 选择 打开 严格 模式 ， 除 
非 你 知道 要 改变 什么 。 参 见 http://dev.mysql.com/doc/refman/5.5/en/server- 
sql-mode.html 了 解 更 多 信息 。 做 出 目 己 的 选择 并 单 击 Next 按 钮 继续 。 


提示 : 


别 筷 了 修改 防火 墙 规则 以 允许 数据 流 从 3306 端 口 〈 或 者 任何 你 确定 用 于 MySQL 的 端口 ) 


11. Networking Options ## H 2 Ja ~Character Set 选 项 。 默 认 的 选项 
古 Standard Character Set， 这 使 得 整个 数据 库 都 采用 Latin1。 也 可 以 选 
择 “Best Support for Multilingualism” 选 项 ， 这 使 得 UTF8 作 为 字符 集 。 
UTF8 人 允许 我 们 在 一 个 单个 字符 集中 存储 多 种 语言 。 如 果 你 想 要 使 用 菏 
个 特定 的 字符 集 ， 选 择 “Manual Selected Default Character Set” 音 选 按 
钮 ， 然 后 从 下 拉 列 表 中 选择 相应 的 字符 集 。 在 做 出 选择 之 后 ， 单 击 Next 
按钮 继续 。 


12. 推荐 把 MySQL 作 为 一 项 服务 安装 。 选 中 “Install as Windows 
Service” 复 选 框 并 且 为 服务 选择 一 个 名 字 。“Launch the MySQL Server 
Automatically” 复 选 框 是 可 选 的 。 还 可 以 选择 把 MySQL bin A aes E! 
Windows PATH， 以 便 更 容易 地 从 命令 提示 从 窗口 调用 MySQL， 如 果 这 
种 情况 很 适合 你 ， 惑 选中 这 个 复 选 枉 。 完 成 了 选择 之 后 ， 单 击 Next 控 钮 


13. Security Options 配 置 界 面 也 是 所 有 配置 界面 中 最 重要 的 一 个 。 
如 图 2-10 所 示 ， 使 用 这 一 配置 窜 面 来 设置 一 个 root 用 户 的 密码 。 输 入 密 
代 两 钦 以 便 确 认 。 不 要 选中 “Enable Root Access From Remote 
Machines” 复 选 框 ， 除 非 你 真 的 知道 在 做 人 什么。 通常 ，root 用 户 连 接 只 人 允 
许 来 目 服 务 器 本 地 。 男 外 ， 可 以 创建 一 个 匿名 用 户 ， 但 出 于 安全 原因 ， 
并 不 建议 这 么 做 。 完 成 了 这 个 寞 面 中 的 配置 选项 之 后 ， 单 击 Next 按 钮 继 


E 
Z5 o 


MySQL Server Instance Configuration Wizard 
MySOL Server Instance Configuration 
Configure the MySQL Server 5.5 server instance. 
Please set the security options. 
[vw Modify Security Settings 
New root password: 
Confirm: 


| Enable root access from remote machines 


. Create An Anonymous Account 


This option will create an anonymous account on this server. 
Please note that this can lead to an insecure system. 





图 2-10 ”通过 MySQL 配 置 ， 为 root 创 建 一 个 密码 


14. 配置 过 程 还 体 留 了 另 一 个 更 多 的 步骤 ， 单 击 Execute 按 钮 可 以 
开始 这 一 步骤 。 在 同 寻 完成 了 各 种 配置 步骤 之 后 ， 将 会 看 到 如 图 2-11 上 所 
示 的 一 个 配置 界面 ， 表 示 配 置 文件 已 经 创建 ， 并 且 MySQL 服 务 已 经 局 
动 。 单 击 Finish 按 钮 结束 问 导 。 


MySQL Server Instance Configuration Wizard 


MySOL Server Instance Configuration 


Configure the MySQL Server 5.5 server instance, 


Processing configuration ... 


iw) Prepare configuration 

iw) Write configuration file (C:\Program Files\Phy SOL tySAL Server 5. 5\my. ini) 
w Start service 

w) Apply security settings 

Configuration file created. 

Windows service MySQL installed. 

Service started successfully. 

Security settings applied. 


Press [Finish] to close the Wizard. 





图 2-11 MySQL 配置 问 导 完成 其 任务 


安装 和 配置 同 导 的 完成 会 产生 一 个 运行 的 MySQL 服 务 ， 并 日 在 
C:\Program Files\MySQL\ MySQL Server 5.5\ 目 录 下 产生 一 个 自 定义 的 
my.ini XF- 


m 
> 


可 以 使 用 任何 文本 编辑 器 来 手动 编辑 my.ini 文 件 ， 在 修改 之 后 必须 重新 局 动 MySQL 服 务 




















现在 MySQL 已 经 启动 了 ， 跳 转 到 2.7 节 。 如 果 在 安装 中 人 磅 到 任何 问 
ol, AA 2.6. 


2.6 ZR EHEER 


OUR TEMySQL ZR FEF EE ee, A 0% a MySQL-F 
册 的 附录 A， 它 位 于 http://dev.mysql.com/doc/refman/5.5/en/problems.html 


e 在 Linux/UNIX 和 Mac OS XX 上 ， 不 正确 的 权限 许可 不 允许 你 启动 
MySQL 守 护 进 程 。 如 果 情 况 是 这 样 ， 确 你 你 已 经 把 所 有 者 和 组 修 
改 为 与 安 疙 说 明 中 相 逻 配 的 那些 内 容 。 

。 如 末 在 连接 到 MySQL 的 时 候 看 到 了 消 居 Access denied， 请 确保 你 使 
用 了 正确 的 用 户 名 和 密码 。 

e 如 采 看 到 了 消息 Can't connect to server， 请 人 确保 MySQL 和 守护 进程 在 


运行 。 


如 果 在 阅读 了 了 MySQL 手册 的 附录 A 之 后 仍然 有 问题 ， 那 么 发 友 邮 件 
下 MySQL 邮件 列表 (参见 http://lists.mysql.com/ 了 解 更 多 信息 ) 可 能 会 
知道 结果 。 你 也 可 以 从 MySQL AB 购 买 支持 协议 。 


2.7 基本 安全 规则 


不 管 是 在 Windows、Linux/UNIX 还 是 Mac OS X EZ MySQL, t 
不 管 是 管理 自己 的 服务 器 还 是 使 用 Intermet 服 务 提供 商 所 提供 的 系统 ， 你 
都 必须 理解 基本 安全 规则 。 如 果 你 通过 Internet 服 务 提 供 丙 访问 
MySQL， 需 要 注意 几 个 服务 器 安全 性 的 方面 。 例 如 ， 一 个 非 root 用 户 不 
能 够 修改 或 绕 过 里 份 验证 。 不 斑 的 是 ， 很 多 Internet 服 务 提供 了 商 对 安全 规 
则 坚 不 在 意 ， 让 他 们 的 客户 戏 露 在 外 ， 并 且 在 很 大 程度 上 ， 他 们 没有 意 
识 到 风险 。 


2.7.1 启动 MySQL 


增强 MySQL 的 安全 性 从 服务 颖 的 局 动 阶段 惑 开 始 了 。 如 果 你 不 是 
服务 亏 的 管理 员 ， 束 不 能 改变 服务 需 的 安全 设置 ， 但 是 ， 你 肯定 可 以 否 
看 服务 器 的 安全 性 ， 并 且 向 Internet 服 务 提 供 商 报告 弱点 。 


如 果 MySQL 安 装 在 Linux/UNIX 或 Mac OS X 上 ， 主 要 关心 的 问题 应 
该 是 MySQL 守 护 程序 的 所 有 者 ， 它 不 应 该 是 root。 把 守护 程序 作为 一 个 
非 root 用 户 的 进程 运行 ， 例 如 mysq] 或 database， 将 会 限制 恶意 用 户 获得 
访问 服务 器 或 者 宪 麻 文件 的 能 


提示 : 





可 以 在 Linux/UNIX 或 Mac OS 久 系统 上 使 用 ps 进程 状态 ) 命令 来 验证 进程 的 所 有 者 。 


如 末 看 人 到 MySQL 在 系统 上 作为 root 用 户 运 行 ， 应 立即 联系 你 的 


Internet 服 务 提 供 商 并 且 提 出 意见 。 如 果 你 是 一 个 服务 需 管 理 员 ， 应 该 作 
为 非 root 用 户 局 动 MySQL 进 程 ， 或 者 在 启动 命令 行 里 指定 首选 的 用 户 
名 。 


# mysqld - -user=non root user name 


例如 ， 如 果 你 想 要 作为 用 户 mysql 运 行 MySQL， 使 用 如 下 命令 。 


# mysqld --user=mysql 


然而 ， 启 动 MySQL 的 推荐 方法 是 通过 MySQL 安 装 的 bin 目 录 中 的 
mysqld_safe 启 动 脚本 来 进行 。 


# bin/mysqld safe --user-=mysql & 
2.7.2 ”增强 MySQL 连 接 的 安全 


你 可 以 以 几 种 不 同 的 方式 连接 到 MySQL 监 视 器 或 者 其 他 的 MySQL 
应 用 程序 ， 每 种 方式 都 有 安全 性 风险 。 如 果 你 的 MySQL 安 装 在 上 自己 的 
工作 站 上 ， 和 那些 必须 使 用 一 个 网 络 链接 连接 到 他 们 的 服务 器 的 用 户 相 
比 ， 你 可 以 少 些 担忧 。 


如 采 MySQL 安 搬 在 工作 站 上 ， 最 大 的 安全 风险 瓯 是 MySQL 监 视 圳 
或 MySQL GUI 管理 工具 没有 关注 到 工作 站 的 局 动 和 运行 。 在 这 种 情况 
下 ， 任 何人 都 可 以 利用 并 删除 数据 ， 插 入 假 的 数据 ， 或 者 关闭 服务 规 。 
如 采 你 必须 让 工作 站 在 一 个 公共 领域 内 体 持 无 监控 状态 ， 驳 使 用 一 个 市 
有 密码 的 屏幕 保护 或 锁定 屏幕 的 机 制 。 


如 果 MySQL 和 安装 在 你 的 网 络 之 外 的 一 个 服务 右上 ， 连 接 的 安全 性 
应 该 受到 关注 。 就 像 任何 通 过 Internet 的 数据 传输 一 样 ， 数 据 可 能 被 截 


Iko WARE EATEN, BOR BTS AAA WOKE TS ER IPE 
信息 。 假 设 未 加 密 传 输 的 数据 是 你 的 MySQL 登 录 信息 ， 一 个 恶意 者 现 
在 就 可 以 伪装 成 你 来 访问 数据 库 了 。 


防止 这 种 情况 发 生 的 一 种 方法 ， 吏 是 通过 一 个 安全 的 链接 〈 例 如 ， 
Secure Shell, EISSH) 来 连接 到 MySQL 。 通 过 SSH， 上 所 有 到 远程 机 需 的 
传输 和 来 日 远程 机 颖 的 传输 部 是 加 符 的 。 类 似 地 ， 如 果 你 使 用 一 个 基于 
Web 的 管理 界面 ， 例 如 phpMyAdmin (参见 http://www.phpmyadmin.net/ 
以 了 解 更 多 信息 ， 注 意 ，phpMyAdmin 在 第 1 革 所 介绍 的 其 于 XAMPP 的 
快速 安 北 中 ， 已 经 安装 过 了 ) ， 或 者 你 的 Internet 服 务 提 供 丙 所 使 用 的 另 
一 种 工具 ， 请 通过 一 个 安全 的 HITP 连 接 来 访问 该 工具 。 


在 下 一 市 中 ， 你 将 会 了 解 到 MySQL 的 权限 系统 ， 这 有 助 于 使 服务 
at RAS TR RRA LER . 


2.8 MySQL 权限 系统 简介 


MySQL 权 限 系 统 忌 是 起 作用 的 。 当 你 第 一 次 笠 试 连接 MySQL 服 务 
俐 的 时 候 ， 并 且 对 于 每 一 个 后 续 的 动作 ，MySQL 都 会 检查 以 下 3 件 事 
情 。 

e 你 从 哪里 访问 《你 的 主机 ) ? 
o 你 说 你 是 谁 〈( 你 的 用 户 名 和 和 密码 〉? 
。 人 允许 你 做 什么 《你 的 命令 权限 ) ? 


所 有 这 些 信息 都 存储 在 一 个 名 为 mysql 的 数据 库 中 ， 当 安装 MySQL 
的 时 候 ， 自 动 创建 该 数据 库 。 在 mysql 数 据 库 中 ， 有 如 下 几 个 和 权限 相 
天 的 表 。 


为 一 个 表 中 的 具体 字段 定义 用 户 权限 。 
。 db 一 一 为 服务 需 上 的 所 有 数据 库 定 义 许可 。 

定义 连接 到 一 个 具体 数据 库 的 、 可 接受 的 主机 。 
为 存储 例 程 定义 用 尸 权限 。 

为 一 个 数据 库 中 的 具体 的 表 定 义 用 户 权 限 。 
为 一 个 具体 用 户 定义 命令 权限 。 


e columns_priv 





e host 








e procs_priv 


e tables_priv 





e user 





在 本 草 中 ， 当 你 同 MySQL 添 加 一 些 示例 用 户 的 时 候 ， 这 些 表 将 变 
得 更 为 重要 。 现 在 ， 只 需要 记 住 这 些 表 的 存在 ， 并 且 为 了 让 用 户 完 成 操 
作 ， 这 些 表 中 必须 拥有 相关 的 数据 。 


2.8.1 两 步 丑 份 验证 过 程 


正如 你 所 了 解 的 ， 在 身份 验证 过 程 中 ，MySQL 检 查 3 件 事情 。 和 这 
3 件 事情 相关 的 动作 分 如 下 两 步 执行 。 


1. MySQL 奏 看 你 的 连接 所 来 目的 主机 ， 以 及 所 使 用 的 用 户 名 和 密 
码 。 如 果 主 机 允许 连接 ， 你 的 用 户 名 对 应 的 密码 正确 ， 并 且 用 户 名 和 分 
配给 该 主机 的 一 个 用 户 名 匹配 ，MySQL 就 转 到 第 二 步 。 


2. 对 于 你 尝试 执行 的 任何 一 条 SQL 命令 ，MVySQL 验 证 你 能 够 对 该 
数据 库 、 表 和 字段 执行 此 操作 。 如 果 步 骤 1 失 败 ， 你 将 会 看 到 一 个 相关 
的 错误 ， 并 且 不 能 继续 步骤 2。 人 例如， 假设 你 使 用 一 个 用 户 名 joe 和 一 个 
密码 abc123 连 接 到 MySQL， 并 且 想 要 访问 一 个 名 为 myDB 的 数据 库 。 如 
果 由 于 如 下 原因 导致 这 些 连接 变量 的 任何 一 个 不 正确 ， 你 都 会 接收 到 一 
条 类 似 如 下 的 错误 消息 。 


。 BSA IEM o 

。 用 户 名 joe 不 存在 。 

。 用 户 joe 不 能 从 localhost 连 接 。 
e 用户 joe 能 够 从 localhost 连 接 ， 但 不 能 使 用 myDB 数 据 库 。 


你 可 能 看 到 如 下 的 一 条 钳 误 消 恩 。 


# mysql -h localhost -u joe -pabci23 test 
Error 1045: Access denied for user: 'joe@localhost' (Using password: YES) 


如 果 市 有 密码 abc123 的 用 户 joe 人 允许 从 localhost 连 接 到 myDB 数 据 
库 ，MySQL 将 会 在 这 个 过 程 的 第 二 个 步骤 中 检查 joe 所 能 执行 的 操作 。 
为 了 便于 说 明 ， 假 设 jow 允 许 伍 询 数 据 但 是 不 允许 插入 数据 。 事 件 和 错 
误 的 序列 就 会 如 下 所 示 。 


# mysql -h localhost -u joe -pabc123 test 

Reading table information for completion of table and column names 
You can turn off this feature to get a quicker startup with -A 
Welcome to the MySQL monitor. Commands end with ; or \g. 

Your MySQL connection id is 12 to server version: 5.5.21-log 
Type 'help;' or '\h' for help. Type '\c' to clear the buffer. 
mysql> SELECT * FROM test table; 

a a ree ~ 

| id | test field | 

PIS BONES ENEMIES 2 一 

| 1 | blah | 

| 2 | blah blah | 

十 ---- 十 ---------- -- 十 


2 rows in set (0.0 sec} 


mysql> INSERT INTO test_tahle VALUES (°'', ‘my text'); 
Error 1044: Access denied for user: ‘joce@localhost' (Using password: YES) 


基于 操作 的 许可 在 具有 多 层级 管理 的 应 用 程序 中 很 常见 。 例 如 ， 如 
果 已 经 创建 了 包含 个 人 财务 数据 的 应 用 程序 ， 你 必须 确保 对 记 账 级 列 的 
Bi ta AWA PSELECTAER, MIRA EY H EE Bee a PP INSERT 
和 DELETE 权 限 。 


在 大 多 数 情况 下 ， 当 你 通过 一 个 Internet 服 务 提供 商 访 问 MySQL 的 
时 候 ， 只 有 一 个 用 户 和 一 个 数据 库 可 供 使 用 。 默 认 情 况 下 ， 一 个 用 户 将 
能 够 访问 该 数据 库 中 所 有 的 表 ， 并 且 允 许 执 行 所 有 的 命令 。 在 这 种 情况 
下 ， 作 为 开发 者 ， 你 的 职 贡 就 古 明 过 目 己 的 编程 开发 出 一 个 安全 的 应 用 
仁厚 。 


然而 ， 如 果 你 是 目 己 的 服务 器 的 管理 员 ， 或 者 Internet 服 务 提 供 丙 多 
许 你 任意 这 加 多 个 数据 库 和 有 用户， 并 且 可 以 修改 上 自己 的 用 户 的 访问 权 
限 ， 下 面 几 个 小 节 将 珊 你 学 习 如 何 做 到 这 些 。 


2.8.2 ”添加 用 户 


通过 一 个 第 三 方 应 用 程序 来 管理 服务 器 ， 这 为 你 提供 了 一 个 简单 的 
方法 来 添加 用 户 ， 只 要 使 用 一 个 类 似 同 导 的 过 程 或 一 个 图 形 化 界面 。 然 
而 ， 通 过 MySQL 监 视 硕 添加 用 户 并 不 难 ， 盛 其 是 如 果 你 理解 了 MySQL 
所 使 用 的 安全 检查 点 ， 这 我 们 刚才 已 经 学 习 过 。 


深 加 新 用 户 的 了 最 简单 的 方法 束 是 便 用 GRANT 命令 。 作 为 root 用 户 连 
接 到 MYSQL， 我 们 融 可 以 使 用 一 条 命令 来 建立 一 个 新 用 户 。 态 一 种 方 
法 是 使 用 INSERT 语 句 修 改 mysql 数 据 库 中 所 有 相关 的 表 ， 这 和 需要 我 们 知 
这 用 来 存储 许可 的 表 的 所 有 字段 。 第 二 种 方法 的 效 末 和 GRANT 命令 相 
同 ， 但 是 比 GRANT 命 令 复 杂 得 多 。GRANT 命 令 的 简单 语法 如 下 。 

GRANT privileges 
ON databasename, tablename 


TO username@host 
IDENTIFIED BY "password"; 


PB fle Ba ARTAS Ee ABI 2 WR ie e BB 
K, We fF http://dev.mysql.com/doc/refman/5.5/en/grant.html 的 
MySQL 手 册 的 GRANT 条 目 。 


ALL 一 一 授予 用 户 所 有 篆 见 权限 。 

ALTER 一 一 用 户 可 以 改变 《修改 ) 表 、 列 和 索引 。 

CREATE 一 一 用 户 可 以 创建 数据 库 和 表 。 

DELETE 一 一 用 户 可 以 从 表 中 删除 记录 。 

DROP 一 一 用 户 可 以 删除 表 和 数据 库 。 

FILE 一 一 用 户 可 以 谈 取 和 写 入 文件 ， 这 个 权限 用 来 导入 或 转 储 数 
据 。 

INDEX 一 一 用 户 可 以 添加 或 删除 索引 。 

INSERT 一 一 用 户 可 以 癌 表 中 添加 记录 。 


PROCESS 一 一 用 户 可 以 得 看 并 俘 止 系统 进程 ， 只 有 可 信任 的 用 户 
才能 拥有 此 权限 。 

RELOAD 一 一 用 户 可 以 使 用 FLUSH 语 句 ， 只 有 可 信任 的 用 户 才能 
拥有 此 权限 。 

SELECT 一 一 用 户 可 以 从 表 中 选取 记录 。 

SHUTDOWN 一 一 用 户 可 以 关闭 MySQL 服 务 器 ， 只 有 可 信任 的 用 
户 才 能 拥有 此 权限 。 

UPDATE 一 一 用 户 可 以 更 新 《修改 ) 表 中 的 记录 。 


例如 ， 如 果 你 想 要 创建 一 个 带 有 99hjc!5 密 码 的 用 户 john， 他 在 名 为 
myDB 的 数据 库 中 的 所 有 表 上 都 有 SELECT 和 INSERT 权 限 ， 并 且 希 望 这 
个 用 户 能 够 从 任何 主机 连接 ， 那 么 ， 使 用 如 下 命令 。 


GRANT SELECT, INSERT 

ON myDB.* 

TO jonn@e"%s" 

IDENTIFIED BY "99hjc!5°; 


FE Tok PAE FFA OY EH. PA EE OR EME TERK 
例子 中 ，* 代 着 了 数据 库 的 全 部 表 ， 而 % 代 苦 了 已 知 的 世界 中 的 所 有 主 
机 的 列表 ， 这 实际 上 是 非常 长 的 一 个 列表 。 


这 里 还 有 使 用 GRANT 命令 添加 用 户 的 另外 一 个 例子 。 这 次 是 琴 加 
一 个 带 有 和 密码 45sdg11 的 用 户 jane， 他 在 名 为 myCompany 的 数据 库 的 一 
个 名 为 employees 的 表 上 具有 ALL 权 限 。 这 个 新 的 用 户 只 能 从 一 个 特定 
的 主机 连接 。 
GRANT ALL 
ON myCompany.employees 


TO jane@janescomputer. company.com 
IDENTIFIED BY "45sdg11"; 


如 果 你 知道 janescomputer.company.com 有 一 个 IP 地 址 63.124.45.2， 
可 以 用 这 个 地 址 来 将 代 命令 中 的 主机 和 名 部 分 ， 命 令 如 下 所 示 。 
GRANT ALL 
ON myCompany.employees 


TO jane@janescomputer. company.com 
IDENTIFIED BY "45sdg11°; 


添加 用 户 的 时 候 需 要 注意 一 点 : 总 是 使 用 密码 并 且 确 保 这 个 密码 是 
不 另外 破解 的 。 


如 果 你 使 用 GRANT 命令 来 添加 用 户 ， 改 变 会 立即 生效 。 要 绝对 确 
保 这 一 点 ， 你 可 以 在 MySQL 监 视 需 中 使 用 一 条 FLUSH PRIVILEGES 命 
令 来 重新 载 入 授权 表 。 

2.8.3” 移 际 权 限 

移 除 权限 和 添加 权限 一 样 简 单 ， 只 不 过 是 使 用 REVOKE 命 令 ， 而 不 
是 使 用 GRANT 命令 。REVOKE 命 令 的 语法 如 下 。 

REVOKE privileges 


ON databasename.tabiename 
FROM username@hostname ; 


我 们 授权 许可 来 使 用 INSERT 命 令 ， 采 用 同样 的 方式 ， 我 们 也 可 以 
通过 使 用 DELETE 命 令 从 mysql 数 据 库 的 表 中 删除 记录 ， 从 而 取消 前 面 
的 授权 许可 。 然 而 ， 这 二 要 你 熟悉 字段 和 表 ， 并 且 使 用 REVOKE 命 令 会 
更 容 匈 和 安全 。 


要 把 用 户 john 同 myCompany 数 据 库 中 的 INSERT 能 力 收 回 ， 可 以 使 
用 如 下 一 条 REVOKE 语 句 。 


| REVOKE INSERE | 


ON myCompany.* 
FROM john@hostname 





对 权限 表 中 的 数据 的 修改 会 立刻 生效 ， 但 是 ， 为 了 让 服务 器 立刻 意 
4 到 你 的 修改 ， 在 MySQL 监 视 器 中 可 以 使 用 FLUSH PRIVILEGES 命 
令 。 


X 


i 


g 


N 


2.9 Zi 


在 Windows 和 Mac OS X 上 安 半 MySQL 是 一 个 简单 的 过 程 ， 因 为 有 
一 个 基于 向 导 的 安装 方法 。LinuxUNIX 用 户 没 有 一 个 基于 向 导 的 安装 过 
程 ， 但 是 ， 按 照 一 组 人 简单 的 命令 来 拆 包 MySQL 客 户 机 和 服务 器 二 进 制 
文件 也 并 不 困难 。Linux/UNIX 用 户 也 可 以 使 用 RPM 来 安装 MySQL。 


安全 性 总 是 首要 的 问题 ， 可 以 采取 几 个 步 又 来 确保 安全 地 安装 
MySQL。 即 便 你 不 是 服务 器 管理 员 ， 还 是 应 该 能 够 认识 到 安全 漏洞 并 
且 提 醒 服 务 器 管理 员 注意 。 


MySQL 服 务 器 不 应 该 作为 root 用 户 运 行 。 此 外 ， 在 MySQL 中 创建 用 
户 时 总 是 应 该 有 一 个 密码 ， 并 且 他 们 获取 的 权限 也 应 该 是 定义 好 的 。 


对 于 每 一 次 请 求 ，MySQL 痢 是 在 两 个 步骤 的 过 程 中 使 用 权限 表 ， 
从 而 知道 你 是 谁 以 及 你 从 哪里 连接 而 来 ， 并 且 信 息 的 每 一 部 分 必须 和 权 
限 表 中 的 一 个 条 目 相 匹配 。 此 外 ， 所 使 用 的 用 尸 员 份 必 须 有 执行 你 所 做 
请 求 的 类 型 的 具体 权限 。 


可 以 使 用 GRANT 命 令 来 洪 加 用 户 权 限 ， 它 使 用 一 个 人 简单 的 语法 来 
加 mysql 数 据 库 中 的 user 表 添加 条 目 。REVOKE 命 令 通 党 也 很 简单 ， 用 来 
移 除 这 些 权 限 。 


2.10 Q&A 


Q: 如 何 完全 删除 一 个 用 户 ? REVOKE 命 令 只 是 删除 了 权 
BRE 。 


: 要 完全 从 权限 表 中 删除 一 个 用 户 ， 必 须 对 mysq] 数 据 库 的 user 和 表 
使 用 wre" 门 的 DELETE 命 令 。 


Q: WRR E ee 了 商 俘 止 把 MySQE 当 作 
root 进 程 运行 ， 而 他 却 不 听 ， 访 怎么 办 ? 


A: 换 一 家 Internet 服 务 提 供 商 。 如 采 你 的 Internet 服 务 提 供 商 没有 认 
识 到 像 把 数据 库 这 样 重 要 的 东西 当 作 root 用 尸 进 程 运行 的 风险 ， 并 且 不 
听取 你 的 要 求 ， 残 另外 找 一 家 提供 商 。 有 些 提 供 商 只 需 每 月 $2.95 的 究 
用 《甚至 是 免费 ) ， 他 们 不 会 将 重要 进程 作为 root 用 户 进 程 运行 。 


211 实践 练习 


这 个 实践 练习 设计 用 来 帮助 你 预测 可 能 的 问题 、 复 习 已 经 学 过 的 知 
识 ， 并 且 开 始 把 知识 用 于 实践 。 


yr] ed 

1. 判断 真 假 : 对 于 从 一 个 远程 主机 到 MySQL 的 安全 连接 ，SSH 是 
完全 可 接受 的 方法 。 
.每 次 同 MySQL 服 务 器 发 出 一 个 请 求 的 时 候 ，MySQL 检 查 哪 3 部 
3. 要 授予 localhost 上 的 名 为 bil 的 用 户 对 于 BilIDB 数 据 库 的 所 有 表 


的 SELECT、INSERT 和 UPDATE 权 限 ， 使 用 什么 命令 ? 另外 ， 为 了 安全 
性 的 考虑 ， 推 荐 的 这 条 语句 中 漏 掉 了 哪些 信息 ? 


oy 


解答 


1. 真 。SSH 将 主机 之 间 的 数据 加 窗 ， 因 此 ， 确 保 了 到 服务 颖 的 连 
接 是 安全 的 。 


2. 你 是 谁 ， 你 从 哪里 访问 ， 允 许 你 执行 什么 操作 。 


=p 


3， 命 令 是 


GRANT SELECT, INSERT, UPDATE 
ON BillDB.* 
TO bill@localhost; 


遗漏 的 重要 信息 是 用 户 的 密码 。 


EA wel 


L 考虑 一 下 你 希望 在 表 级 限制 访问 命令 的 情况 。 例 如 ， 你 不 希望 
实习 级 别 的 管理 员 拥有 关闭 公司 数据 库 的 权限 。 


2. 如 果 你 在 MySQL 中 有 管理 员 权 限 ， 使 用 几 条 GRANT 命令 创建 
一 些 假 的 用 户 。 你 所 指定 的 表 和 数据 库 实 际 上 是 否 存 在 无 关 紧 要 。 


3. 使 用 REVOKE 来 移 除 在 思考 题 2 中 所 创建 的 用 户 的 茶 些 权限 。 


Set 2232 AAC Apache 


在 本 章 中 ， 你 将 学 习 : 


e 如 何 安装 Apache 服 务 器 。 
e 如 何 对 Apache 做 配置 修改 。 
。 Apache 的 日 志和 配置 文件 存储 在 何 处 。 


EREE, RIZ Apache Web 服 务 占 并 熟悉 它 的 主要 组 件 ， 包 
括 日 志和 配置 文件 。 


3.1 Apache 的 当前 版 本 及 未 来 版 本 


当 你 访问 位 于 http://httpd.apache.org 的 Apache HITPD 服 务 需 站 点 ， 
将 会 看 到 关于 Apache 2.0.x. Apache 2.2.x 和 Apache 2.4.x 版 本 的 发 布 声 
明 。 正 如 根据 版 本 号 可 以 猜 到 的 ，Apache 2.0.x 是 这 三 个 版 本 中 最 老 的 
版 本 。Apache 软 件 基金 会 (Apache Software Foundation) 维护 了 所 有 这 
3 个 版 本 ， 但 是 ，Apache 2.4.x 的 功能 包含 了 对 过 滤 、 绥 存 、 负 载 均 衡 以 
及 其 他 系统 功能 很 好 的 文 持 。 这 也 是 本 草 中 使 用 的 版 本 。 然 而 ， 无 论 你 
it FE ZI Apache 2.2.x 还 是 Apache 2.0.x， 本 书 中 所 有 的 PHP 和 MySQL 代 
但 都 能 够 像 摘 述 的 那样 工作 。 实 际 上 ， 你 将 会 用 现 ， 相 当 多 的 托管 提供 
商 仍 然 使 用 服务 右 的 Apache 2.0.x 碑 本 ， 而 不 是 Apache 2.2.xX， 更 不 要 说 
最 新 的 Apache 2.4.x 版 本 了 。 如 果 你 在 按照 本 书 介 绍 的 内 容 安 装 Apache 
2.4.X 的 时 候 有 任何 问题 ， 请 答 试 安 疼 其 前 一 个 版 本 《例如 ，Apache 
2.2.x) ; 安 效 说 明 非 党 相似 。 本 重 的 安 效 说 明 针 对 Apache HITPD 服 务 
石上 的 2.4.1 碑 ， 这 是 在 编 与 本 书 的 时 候 可 获得 的 该 软件 的 最 好 厂 本 。 


Apache 软 件 基 金 会 (Apache Software Foundation) 对 包 侣 增强 安全 
或 问题 修复 的 升级 使 用 次 版 本 号 。 识 版 本 没有 预定 的 用 布 计 划 ， 当 对 代 
人 码 进 行 了 扩展 和 修复 并 且 进 行 了 彻 压 的 测试 后 ，Apache 软 件 基金 会 束 会 
用 一 个 新 的 次 版 本 号 来 及 布 一 个 新 的 版 本 。 


当 你 购买 本 书 时 ， 有 可 能 次 版 本 写 已 经 友 生 改变 ， 变 为 2.4.1 或 更 
高 。 如 果 是 这 种 情况 ， 你 应 该 阅读 改变 的 列表 ， 从 
http://httpd.apache.org/download.cgi 的 下 载 区 可 以 找到 这 个 列表 的 链接 ， 
其 中 所 有 关于 安装 或 配置 过 程 的 改变 组 成 了 变更 列表 的 大 部 分 内 容 。 


里 然 在 次 版 本 升级 中 不 太 可 能 所 有 的 安 北 说 明 儿 改变， 但 是 应 设 弄 
成 她 终 检 权 目 己 安 农 和 维护 的 软件 的 改变 日 志 的 习惯 。 如 果 在 你 阅读 本 
书 期 间 次 厂 本 及 生 了 改变 ， 但 是 在 改变 日 专 中 没有 新 的 安 冯 改变 记录 ， 
你 只 要 用 心 记 下 新 的 版 本 号 ， 并 且 当 版 本 写 逢 要 出 现在 安 疙 说 明和 相关 
的 图 中 的 时 候 ， 用 最 新 版 本 写 荃 换 束 行 了 。 


3.2 ”选择 合适 的 安装 方法 


当 安 装 过 程 进行 到 适当 的 位 置 ， 并 获得 一 个 基本 的 Apache 安 闭 时 ， 
你 有 几 个 选择 。Apache 是 开源 软件 ， 这 意味 看 你 有 权 使 用 软件 的 全 部 源 
代码 ， 也 束 是 允许 你 编 详 目 己 目 定义 的 服务 器 。 帮 外， 预先 编译 好 的 
Apache 二 进 制 代码 友 布 对 大 多 数 现代 UNIX 平 台 部 可 用 。 取 后 ，Apache 
己 经 捆绑 到 很 多 种 Linux 发 布 上 ， 你 还 可 以 从 软件 供应 商 那里 购买 带 有 
支持 包 的 商业 版 。 如 果 使 用 的 是 Linux/UNIX， 本 章 中 的 示例 将 教 你 如 何 
WARS ZA Apache; 如 果 你 计划 在 Windows 系 统 上 运行 Apache， 本 章 
中 的 示例 将 教 你 如 何 使 用 安 浅 程序 。 


3.2.1 ”从 源 代 人 码 安 装 


从 源 代 人 码 安装 将 禹 给 你 最 大 的 灵活 性 ， 因 为 它 允 许 你 安装 一 个 目 定 
义 服务 器 ， 移 除 不 需要 的 模块 ， 并 有 旦 用 第 三 方 模块 扩展 服务 器 。 从 源 代 
人 码 安 装 Apache 使 你 轻松 升级 到 最 狐 版 本 并 快速 应 用 安全 补丁 ， 而 由 供应 
丙 提供 的 最 新 版 本 可 能 要 数 天 甚至 数 周 后 才能 发 布 。 对 于 简单 安装 ， 从 
源 代码 安 狼 Apache 的 过 程 并 不 是 特别 难 ， 但 是 当 包 括 第 三 方 模块 和 库 
时 ， 安 猴 过 程 的 复杂 上 度 将 有 所 增加 。 
3.2.2 ”安装 一 个 二 进 制 代码 版 本 

Linux/UNIX 二 进 制 安 闭 版 本 可 以 从 供应 商 获 得 ， 也 可 以 从 Apache 
软件 基金 会 网 站 下 载 。 针 对 只 有 有 限 的 系统 官 理 知识 的 用 户 或 者 没有 特 
列 配 置 知 要 的 有 用户 ， 他 们 提供 了 一 种 简便 的 安溪 Apache 的 方法 。 第 三 方 
商业 供应 商 提 供 一 个 增加 了 应 用 程序 服务 右 、 附 加 的 模块 、 文 持 等 预 打 


AH Apache 48h As. Apache ERa AWindows #2 he tk [TSB 
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3.3 在 LinuxwUNIX 上 安装 Apache 


这 部 分 介绍 了 如 何在 Linux/UNIX 上 安装 Apache 2.4.1 的 最 新 版 本 。 
成 功 地 从 源 代码 安装 Apache 的 一 般 步 又 如 下 。 


1. 下 载 软件 。 

2. 运行 配置 脚本 。 

3. 编译 代码 并 安装 它 。 

下 面 的 部 分 将 详细 介绍 这 些 步 又 。 
3.3.1 下 载 Apache 源 代码 


官方 的 Apache 下 载 站 点 是 http://httpd.apache.org/download.cgi 。 你 可 
以 找到 Apache 源 代码 的 几 个 版 本 ， 它 们 是 用 不 同 压缩 方法 打包 的 。 友 布 
文件 首 乞 用 tar 应 用 程序 打包 ， 然 后 用 gzip 应 用 程序 或 compress 应 用 程序 
压缩 。 如 果 你 已 经 在 目 己 的 系统 上 安装 了 gunzip 应 用 程序 ， 那 么 下 载 
*.tar.gz 版 本 。 这 个 应 用 程序 在 诸如 FreeBSD 和 Linux 的 开放 源 代码 操作 系 
统 中 己 经 默认 安装 。 如 果 你 的 系统 上 没有 安装 gunzip， 那 么 下 载 *.tar.Z 
文件 (gunzip 并 没有 包含 在 多 数 商 业 UNIX 操 作 系 统 的 默认 安装 中 )。 


你 想 下 载 的 文件 将 命名 为 类 似 httpd-VERSION .tar.gz 的 文件 ， 其 
H, VERSION Apachel it 275K AS. WGN, Apache 2.2.8 下 载 并 
存 成 一 个 名 为 httpd-2.4.1.tar.gz 的 文件 。 把 下 载 文件 放 到 为 源 文 件 保留 的 
目录 ， 例 如 /usr/src/ 或 /usr/local/src/。 


3.3.2 ”解压 源 代 公 


如 果 下 载 了 用 gzip 压 缩 的 tarball (A? tar.gz) ， 可 以 用 gunzip 
应 用 程序 (gzip 友 布 的 一 部 分 ) KEEA 


提示 : 


tarball 和 是 对 使 用 tar 应 用 程序 压缩 软件 的 一 个 通用 别名 。 


你 可 以 通过 输入 如 下 命令 来 解压 缩 并 拆 包 软件 。 


# gunzip < httpd-2.4*.tar.gz | tar xvf - 


ff Jk Agtarball fy E S—S BZA, TA 4 httpd VERSION 的 顶级 
目录 。 把 当前 目录 切换 到 这 个 顶级 目录 ， 准 备 配 置 软件 。 


3.3.3 ER iE Apache 


你 可 以 通过 使 用 在 顶级 发布 目录 中 的 configure 脚 本 ， 指 定 生成 的 二 
进 制 代 人 码 中 将 有 哪些 特性 。 献 认 情 况 下 ， 把 Apache 编 详 为 一 系列 前 态 编 
译 的 标准 模块 ， 并 安装 在 /usr/local/apache2 目 录 。 如 果 乐 于 使 用 这 些 设 
置 ， 可 以 输入 如 下 命令 来 配置 Apache。 


# ,/configure 


然而 ， 为 给 第 4 草 中 PHP 安 冯 做 准备 ， 你 需要 确 你 把 mod_so 模 块 编 
译 进 Apache。 以 UNIX 共 于 对象 (*.s0) 格 式 命 名 的 这 个 模块 ， 可 以 让 
Apache 激 活 诸如 PHP 这 样 的 动态 模块 。 在 特定 位 置 〈 在 本 例 中 
是 /usr/local/apache2/) 配置 Apache 来 安装 它 日 己 ， 并 允许 使 用 mod_so， 
得 入 如 下 命令 。 


# ./configure --prefix=/usr/local/apache2 --enable-so 


configure 脚 本 的 目的 是 解决 关于 奏 找 库 、 编 译 时 选项 、 特 定 乎 台 郑 
弄 等 所 有 的 事情 ， 并 创建 一 系列 叫做 makefiles 的 专用 文件 。makefiles 包 
含 执行 不 同 任务 〈 例 如 安装 Apache) 的 指令 ， 叫 做 targets。 这 些 文 件 由 
make 必 用 程序 来 谈 取 ， 它 将 执行 这 些 任务 。 如 有 果 一 切 有 顺利， 执行 
configure 后 你 将 看 到 一 系列 和 刚才 执行 的 不 同 检 杏 相关 的 消 轧 ， 并 将 返 
回 到 命令 提示 行 。 


configure ok 


creating test/Makefile 
config.status: creating docs/cont/httpd.contf 


config.status: executing default commands 
# 


gn RconfigurefJA AM, KERE, JEEE BWI z E p He 
ARTE, DUm ENSE. ERNA RZD EE AI H SC BR 
config.log#iconfig.status XF Ja, VRP UAA configure + - 


WR AC Bee AG th EB, MRA ERAPR, HBA, Vi 
问 httpd://apr.apache.org/ 并 且 下 载 APR 和 APR-util 软 件 包 ， 并 且 将 它们 解 
AREN Aa VERSIONI H aeWsrclibf HKF. ~H wi sen, 
再 次 运行 配置 命令 。 类 似 的 ， 如 采 配 置 过 程 最 终 给 出 一 个 警告 ， 说 你 没 
有 安装 PCRE， 访 问 http:/www.pcre.org 并 下 载 该 文件 ， 根 据 该 Web 站 点 
的 说 明 在 你 的 系统 上 安 痛 它 。 安 痛 完 成 后 ， 再 次 运行 配置 命令 。 


和 Apache 2.2.X 的 安 儿 过 程 不 同 ， 在 Apache 2.4.x ZANTE, X 
两 项 需求 都 会 有 所 改变 。 


3.3.4 ni Fl 228 Apache 


make 应 用 程序 读 取 保存 在 makefiles 中 的 信息 并 编译 服务 器 和 模块 。 
在 命令 行 输入 make 命 令 以 编译 Apache。 你 将 看 到 指示 编译 的 进程 的 几 
个 消息 ， 最 后 将 回 到 命令 提示 行 。 编 译 结束 后 ， 你 可 以 通过 在 命令 提示 
行 中 输入 make install 来 安装 Apache。makefiles 将 安装 文件 和 目录 ， 并 返 
器 到 命令 所 示 行 。 
rista tiing header files 
Installing build system files 
Installing man pages and online manual 


make[1]: Leaving directory | /usr/local/bin/httpd-2.4.1' 
# 


正如 configure 命 令 中 的 --prefix 开 关 指 定 的 那样 ，Apache 发 布 文 件 现 
在 将 在 /usr/local/apache2 目 杂 中 。 有 要 测试 httpd 二 进 制 代 人 码 是 人 耕 已 经 正确 
编译 ， 在 命令 提示 行 输 入 如 下 命令 。 
# fusr/local/apache2/bin/httpd -v 


你 将 看 到 如 下 输出 〈( 你 的 版 本 和 编译 日 期 将 有 所 不 同 〉。 


Server version: Apache/2.4.1 (Unix) 
Server built: March 12 2012 11:47:22 


除非 你 想 学 习 如 何在 Mac OS X 或 Windows 上 安装 Apache， 人 个 则 请 
直接 跳 到 “Apache 配 置 文 件 结构 ”一 节 学 习 Apache 的 配置 文件 。 


3.4 在 Mac OS X |. <28Apache 


KIRS, Apache 0% 238 Fl|Mac OS XF. Eki tel F, Apache 
服务 器 二 进 制 代码 位 于 /usrsbinhttpd。 诸 如 httpd.conf 等 配置 文件 ， 即 
Apache 的 主要 配置 文件 ， 保 存在 /etc/httpd。 因 为 Apache 己 经 安装 并 且 完 
全 准备 好 使 用 PHP， 请 直接 跳 到 “Apache 配 置 文件 结构 ”部 分 学 习 Apache 
配置 文件 以 及 如 何 使 用 它 。 


vy. ae 
VER: 





如 果 你 想 要 使 用 针对 Mac OS X 的 一 次 性 软件 包 ， 可 以 像 第 1 章 中 所 介绍 的 那样 使 用 
XAMPP， 或 者 你 可 以 安装 来 自 http:/www.mamp.info 的 MAMP 软 件 包 。 


3.5 7 Windows E %7% Apache 


TREN 
VERS: 


在 编写 本 书 时 ，Apache 2.4.x 对 于 Windows 来 说 还 不 可 用 。 因 此 ， 如 下 的 说 明 是 针对 2.2.x 
的 。 当 Apache 2.4.x 变 得 可 用 的 时 候 ， 其 安装 过 程 将 会 是 类 似 的 。 


Apache 2.2 可 以 在 绝 大 多 数 Windows 平 台 上 运行 ， 并 且 提 供 比 
Apache 2.0 和 Apache 1.3 版 本 更 佳 的 性 能 和 稳定 性 。 你 可 以 从 源 代 码 编译 
Apache， 但 是 因为 不 是 多 数 Windows 用 户 都 有 编译 器 ， 本 节 用 MSI 安 装 
程序 版 本 进行 安装 。 


在 安装 Apache 之 前 ， 你 大 概 想 确 定 当 前 在 目 己 的 机 右上 是 否 已 经 运 
行 了 一 个 Web 服 务 右 “例如 ，Apache 的 前 一 个 版 本 、Microsoft Internet 
Information Server 或 Microsoft Personal Web Server) ， 可 能 希望 秋 载 或 
者 禁用 已 存在 的 服务 右 。 你 可 以 同时 运行 几 个 Web 服 务 絮 ， 但 是 它们 必 
须 运 行 在 不 同 的 地 址 和 刀口 组 合 上 。 


在 下 载 安 斤 程序 之 前 ， 化 一 点 时 间 《 这 是 一 个 非 名 重要 的 时 刻 ) 在 
下 载 页 面 〈 位 于 http:Whttpd.apache.org/download.cgi ) 查找 一 条 人 句子， 内 
容 是 “If you are downloading the Win32 distribution, please read these 
important notes.” o 指 问 这 些 要 点 的 URL 


Je http://www.apache.org/dist/httpd/binaries/win32/README.html 。 


Apache 软 件 基 金 会 维护 这 个 页 面 是 为 了 那些 想 运 行 Apache 服 务 器 版 
本 的 Windows 用 户 。 在 这 个 页 面 里 几乎 有 每 一 个 目前 仍 在 使 用 的 


Windows 版 本 的 要 点 ， 同 样 地 ， 你 最 好 提起 兴趣 阅读 一 下 这 里 的 信息 。 
如 果 你 正在 运行 Apache， 无 论 作 为 一 个 产品 还 是 开发 服务 器， 我 保证 你 
可 以 在 这 个 要 点 页 面 找 到 相关 的 信息 。 


当 你 已 经 准备 好 安 猴 时 ， 答 找 名 为 Win32 Binary (MSI 安 装 程序 ) 
MEER. “PRTC IR, WS MAIER. IMA 
一 个 欢迎 界面 ， 如 网 3-1 所 示 。 


ie Apache HTTP Server 2.2 - Installation Wizard 


Welcome to the Installation Wizard for 
Apache HTIP Server 2.2.22 


The Installation Wizard will install Apache HTTP Server 2.2.22 
on your computer, To continue, dick Next, 


WARNING: This program is protected by copyright law and 
international treaties. 








图 3-1 Windows 安 装 程序 欢迎 界面 


点 击 Next 按 钮 继续 安 朔 过程， 将 提示 你 接 党 Apache 授 权 许 可 。 授 权 
许可 主要 表示 : 除了 声明 是 你 编写 的 这 个 软件 之 外 ， 你 可 以 用 这 个 软件 
做 任何 想 做 的 事 一 一 包括 做 私人 的 修改 。 但 是 一 定 要 阅读 这 个 授权 许 
可 ， 以 便 你 完全 理解 这 些 条 于。 


ZIA AY Za, ERE Za tH —“Apachelt) fai FEIT ZA. ABE 
看 ， 它 要 求 提 供 你 的 计算 机 的 基本 信息 ， 如 图 3-2 所 示 。 这 包括 服务 器 
的 完整 的 网 络 地 址 《举例 来 说 ，mycomputer. mydomain.com) 和 管理 员 
的 Email 地 址 。 服 务 吉 名 是 客户 疾 将 用 来 访问 服务 夯 的 名 字 ， 而 管理 员 
的 E-mail 地 址 将 被 添 加 到 铬 误 信 息 中 ， 以 便于 在 出 现 错误 时 访问 者 知道 
如 何 联 系 你 。 


ie Apache HTTP Server 2.2 - Installation Wizard 


Server Information 


Please enter your server's information, 


Network Domain {e.g. somenet. com) 
| 


Server Name (e.g. www.somenet.com): 


| 


Administrator's Email Address (e.g. webmaster @somenet.com): 


一 一 


Install Apache HTTP Server 2.2 programs and shortcuts for: 


@ for All Users, on Port 80, as a Service -- Recommended, 
O only for the Current User, on Port 8080, when started Manually. 


InstallShield 


[<Back JU Next> 





图 3-2 基本 信息 界面 


这 个 界面 还 将 提示 选择 安 痛 哪 种 局 动 方式 一 一 是 将 Apache 作 为 服务 
启动 还 是 手动 启动 Apache。 把 Apache 作 为 服务 安装 ，Apache 将 在 
Windows 每 次 局 动 时 目 动 运行 ， 并 且 你 可 以 通过 标准 Windows 服 务 管 理 
工具 控制 它 。 为 当前 用 户 安 装 Apache， 需 要 手动 后 动 Apache 并 且 设 置 默 
认 端 口 ，Apache 在 这 个 端口 上 监听 对 8080《〈 而 不 是 80) 的 请求 。 选 择 合 


Jer FA) FPL FEF A tr Next] 4H ARE 


提示 : 


如 果 你 的 机 器 没有 完整 的 网 络 地 址 ， 使 用 localhost 或 127.0.0.1 作 为 服务 器 名 。 


下 一 个 界面 让 你 选择 安装 的 类 型 ， 典 型 typical) AEX 
(custom) 。 上 典型 安装 (Typical installation〉 意 味 看 将 安装 Apache 二 进 
制 代 码 和 文档 ， 但 是 标 头 和 库 将 不 安 疼 。 这 有 是 供 选 择 的 最 佳 选项 ， 除 非 
你 准备 编译 目 己 的 模块 。 


目 定义 安装 (Custom installation) veh (Rept ce A pk F~ 
档 。 选 择 目 标 安装 目录 后 ， 默 认 是 C:\Program Files\Apache Software 
Foundation/Apache 2.2， 程 友 将 处 理 安装 过 程 。 如 琳 一 切 顺 利 ， 它 将 显 
未 一 个 如 图 3-3 所 示 的 最 终 界 和 面 。 


ie Apache HTTP Server 2.2 - Installation Wizard 


Installation Wizard Completed 


The Installation Wizard has successfully installed Apache HTTP 
Server 2.2.22. Click Finish to exit the wizard. 


图 3-3 RD ZR 


在 下 一 节 里 ， 你 将 学 习 有 关 Apache 配 置 文件 的 内 容 ， 并 最 
的 新 服务 需 。 


ne 


Pe 





~ 
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3.6 Apache 配置 文件 结构 


Apache 把 它 所 有 的 配置 信息 都 保存 到 文本 文件 中 。 主 文件 叫做 
httpd.conf。 这 个 文件 包含 指令 和 容 保 ， 它 们 允许 定制 Apache 安 沽 。 指 令 
(Directive) 配 置 Apache 的 特定 设置 ， 例 如 授权 、 性 能 和 网 络 参 数 。 容 器 
(Container) 指定 这 些 设置 作用 的 上 下 文 。 例 如 ， 授 权 配 置 可 以 作用 于 
作为 一 个 整体 的 服务 右 、 一 个 目录 或 单个 文件 。 


3.6.1 指令 
下 面 的 规则 适用 于 Apache 指 令 语 法 。 


指令 参数 跟 在 指令 名 后 面 。 

HOB BUH te aT BATE o 

BRN Be MAA Rata SY AMA, ARETE ORAS A 

— Mat SALAH 477, {Ae feta SB 17 2a Eh 
RALF \) ， 可 以 在 万 一 行 继续 这 个 指令 。 

HS CG) 要 在 指令 之 前 ， 且 必须 每 行 一 个 。 


Apache 服 务 器 文档 
在 http://httpd.apache.org/docs/2.2/mod/quickreference.html 提供 了 一 个 关 
于 指令 的 快速 参考 。 虽 然 和 后 将 学 习 一 些 基 本 的 指令 ， 但 是 你 应 该 使 用 
联机 文档 补 元 目 己 的 知识 。 


和 令 的 Apache 文 档 退 第 都 这 循 如 下 格式 。 


e Description 这 个 条 目 提 供 了 指令 的 一 个 简单 说 明 。 





这 个 条 目 解 释 了 指令 选项 的 格式 。 必 和 需 的 参数 以 科 体 显 
不， 可 选 参 数 以 冬 体 显示 并 用 括号 括 起 来 。 

。 Default 一 一 如 各 指令 有 一 个 默认 值 ， 将 在 这 里 显示 它 。 

这 个 条 目 详细 介绍 了 可 以 显示 指令 的 容 髓 或 区 域 。 容 

句 将 在 下 一 小 节 介 绍 。 可 能 的 值 是 server config. virtual hosts 


e Syntax 








e Context 


directory fil .htaccess . 

Apache 的 指令 属于 不 同 的 种 类 。Override 字 段 用 于 指 

定 哪个 指令 种 类 可 以 显示 在 .htaccess per-directory 配 置 文件 中 。 

这 个 条 目 指明 指令 是 否 在 Apache(core) 中 编译 ， 是 否 属 
于 绑 定 模块 之 一 《base 或 extension， 取 决 于 默认 情况 下 是 否 编 译 它 
们 ) ， 是 多 进程 模块 (Multi-Processing Module，MPM) 的 一 部 分 ， 
还 是 用 Apache 绑 定 但 不 准备 用 在 产品 服务 磊 (experimental〉 中 。 

e Module 这 个 条 目 指明 指令 属于 哪个 模块 。 

Compatibility 一 一 这 个 条 目 包含 有 天 文 持 这 个 指令 的 Apache 版 本 

的 信息 。 


e override 





e Status 








文档 中 的 这 些 项 目 可 以 找到 指令 更 深入 的 解释 ， 并 且 有 天 指令 或 文 
档 的 参考 资料 可 能 蛙 示 在 最 后 。 


3.6.2 ”容器 


fa Bas, TAYE? (section，， 限 制 指令 作用 的 作用 域 。 指 令 如 
果 没 有 在 一 个 容 右 内 ， 则 属于 默认 服务 占 作 用 域 (server config) , FF 
有 昌 作 为 一 个 整体 作用 于 服务 絮 。 


如 下 是 Apache 默 认 有 的 指令 容 右 。 


e <VirtualHost> VirtualHost#s8 S 48 Ee —“ FURS 4s. Apache 
TOE AK A BAS es J Apacheia {7 7S [A] Webi i. IX SA aE NT 
令 作 用 于 一 个 特定 的 Web 站 点 。 这 个 指令 接受 域名 或 1P 地 址 以 及 一 
个 可 选 的 问 口 作为 参数 。 你 将 在 第 29 草 学 习 天 于 虚拟 主机 更 多 的 内 
容 。 


e <Directory>, <DirectoryMatch> 








这 些 容器 允许 指令 作用 于 某 一 
目录 或 者 文件 系统 中 的 目录 组 。Directory 容 器 获取 一 个 目录 或 目录 
结构 参数 。 容 器 包括 的 指令 作用 于 指定 目录 以 及 它们 的 子 目录 。 
DirectoryMatch 容 右 人 允许 指 定 正 则 表达 式 结 构 作为 参数 。 淮 例 来 
说 ， 下 列 内 容 多 许 www 目 录 的 所 有 二 级 子 目录 的 一 个 匹配 ， 它 由 4 
个 数字 组 成 ， 例 如 一 个 按 年 和 月 (0212 代 表 2012 年 2 月 ) 命名 的 目 
SKK 


<DirectoryMatch °*/www/.*/[@-9]{4}°> 


e <Location>, <LocationMatch> 这 些 容 器 人 允许 指令 作用 于 某 个 
请 求 的 URL 或 URL 结 构 。 它 们 与 <Directory>, <DirectoryMatch> 类 
似 。LocationMatch 获 取 一 个 正则 表达 式 作 为 参数 。 例 如 ， 下 列 内 容 
下 配 包含 /my/data” 或 “/your/data” 的 目录 。 





<LocationMatch "/(my|your})/data"> 


e <Files>, <FilesMatch> ZB (WF Directory fU Location h] 44s, Files 
段 允 许 目录 作用 于 地 个 文件 或 文件 结构 。 





容器 包含 指令 ， 如 程序 清单 3.1 所 示 。 


程序 清单 3.1 示例 容器 指令 


: <Directory "/some/directory"> 
: SomeDirectivet 

SomeDirective2 

: </Directory> 

: <Location "/downloads/*.html"> 
: SomeDirectives 

' </Location> 

= Files "4. (gif ipa) > 

: SomeDirective4 

: </Files> 


名 DC noah OMN 一 


— 


示例 指令 SomeDirectivel 和 SomeDirective2 将 作用 于 /some/directory 
日 录 及 其 子 日 录 。 指 令 SomeDirective3 将 作用 于 指 癌 市 有 .html 扩 展 名 的 
页 面 的 URL， 这 个 页 面 位 于 /downloads/URL 下 。SomeDirective4 将 作用 
于 带 有 .gif 或 .jpg 扩展 名 的 所 有 文件 。 


3.6.3 条件 评 估 


Apache 提 供 对 条 件 容器 的 支持 。 只 有 当 某 些 条 件 符合 时 ， 才 执行 这 


些 容 器 中 的 封装 指令 (directives enclosed) 。 


o <IfDefine> 如 采 把 一 个 指定 的 命令 行 开 关 传 递 给 Apache 的 可 执 
行程 序 ， 将 执行 这 个 容 需 内 的 指令 。 只 有 当 把 -DMyModule 开 天 传 
递 给 将 要 执行 的 Apache 二 进 制 代码 时 ， 才 会 执行 程序 清单 3.2 中 的 
指令 。 可 以 直接 在 命令 行 传递 这 个 开关 ， 也 可 以 通过 修改 apachectl 
脚本 来 传递 这 个 开关 ， 稍 后 将 在 本 章 的 "Apache 相关 命令 ”一 下 介 


绍 。 





IfDefine 容 器 也 接受 求 反 的 参数 。 也 就 是 说 ， 只 有 非 -DMyModule 参 
数 作为 命令 行 参数 传递 时 ，<IfDefine IMyModule> 段 中 的 指令 (注意 叹 
号 在 MyModule 名 字 之 前 ) 才 执 行 。 


1: 


=e Be 


—l 


3.6.4 ”ServerRoot 指 令 


e <IfModule> 


Se OO nr DO 上 PP 一 





只 有 当 作 为 参数 传递 的 模块 出 现在 Web 服 务 融 中 


时 ，IfModule 段 中 的 指令 才 会 执行 。 举 例 来 说 ，Apache 载 入 一 个 对 
不 同 MPM 提 供 支 持 的 默认 httpd.conf 配 置 文件 。 正 如 我 们 在 程序 清 

早 3.3 中 见 到 的 那样 ， 只 有 属于 编译 到 Apache 中 的 MPM 的 配置 才 会 
执行 。 这 个 示例 的 目的 是 阐明 将 只 执行 指令 组 中 的 一 个 指令 。 


<IfDefine MyModule> 
2: LoadModule my_module modules/libmymodule.so 
</IfDefine> 


: <IfModule prefork.c> 


: Startservers O 
: MinSpareServers 5 
: MaxSpareServers 10 
: MaxClients 20 
: MaxRequestsPerChild @ 
: </IfModule> 


: <IfModule worker.c> 


: StartServers 3 
: MaxClients 8 
: MinSpareThreads 5 
: MaxSpareThreads 10 
: ThreadsPerChild 25 
: MaxRequestsPerChild @ 
: </IfModule> 


程序 清单 3.2 ”IfDefine 示 例 


程序 清单 3.3”IfModule 示 例 


ServerRoot 指 令 获 取 一 个 单个 的 参数 : 一 个 指 回 服务 器 作用 目录 的 
目录 路 径 。 其 他 指令 中 的 所 有 相对 路 径 引 用 都 是 相对 于 ServerRoot 的 
值 。 如 果 你 在 Linux/UNIX 上 从 源 代 人 码 编译 Apache， 如 本 章 表 和 面 所 述 ， 
ServerRoot 的 默认 值 是 /usr/local/apache2。 对 于 Mac OS XHP, 


ServerRoot 的 默认 值 是 /Library/WebServer。 如 果 使 用 的 是 Windows 安 装 
程序 ，ServerRoot 的 默认 值 是 C:\Program Files\Apache Software 
Foundation\Apache 2.2\. 


3.6.5 ”per-directory 配 置 文件 


Apache 使 用 per-directory 配 置 文件 来 允许 指令 在 主 配 置 文件 http.conf 
之 外 存在 。 这 些 专 用 文件 可 以 存放 到 文件 系统 中 。 如 有 果 请 求 一 个 文档 ， 
而 该 文档 保存 在 包含 上 述 专 用 文件 之 一 的 一 个 目录 中 或 者 它 下 面 的 任何 
子 目 录 中 ，Apache 将 处 理 这 些 专 用 文件 的 内 容 。 所 有 适用 有 的 per- 
directory 配 置 文件 的 内 容 将 会 合并 人 处理。 例如 ， 如 果 Apache 接 收 到 一 个 
对 /usr/local/ apache2/htdocs/index.html StF Wis HK, “EDK 
#£/, /usr. /usr/local. usr/local/apache?2 #il/usr/local/apache2/htdocs H 3k 
中 ， 按 照 顺序 查找 per-directory 配 置 文件 。 


js SS 
YER: 


启用 per-directory 配 置 文件 会 有 性 能 损失 。Apache 必 须 执行 大 量 磁盘 操作 来 查找 每 个 请 求 
中 的 这 些 文 件 ， 即 使 这 些 文 件 不 存在 。 


默认 把 per-directory 配 置 文件 称 为 .htaccess， 这 样 叫 法 是 有 历史 原 
的 。 它 们 最 初 用 于 保护 对 包含 HTML 文 件 的 目录 的 访问 。 


AccessFileName 指 令 人 允许 你 把 per-directory 配 置 文件 名 从 .htaccess 改 
为 其 他 的 名 字 。 它 接受 一 个 文件 名 列表 ， 当 查找 per-directory 配 置 文件 
时 Apache 将 使 用 该 列表 。 


要 确定 一 条 指令 是 否 可 以 在 per-directory 配 置 文件 中 有 履 盖 ， 检 查 指 


令 语 法 定义 的 Context: 字 段 是 个 包含 .htaccess。Apache 指 令 属 于 不 同 的 
组 ， 就 像 在 指令 语法 说 明 中 的 Override 字 段 中 指定 的 那样 。Override 字 上 段 
可 能 的 值 如 下 。 





e AuthConfig 授权 指令 。 














e FileInfo 控制 文档 类 型 的 指令 。 

e Indexes 控制 目录 索引 的 指令 。 

e Limit 控制 主机 访问 的 指令 。 

e Options 控制 指定 目录 特性 的 指令 。 


通过 使 用 AllowOverride 指 令 ， 可 以 控制 哪个 指令 组 能 够 显示 在 per- 
directory 配 置 文件 中 。AllowOverride 也 能 获取 一 个 Al 或 None 参 数 。Al 
意味 看 属于 所 有 有 组 的 指令 痢 能 显示 在 配置 文件 中 。None 意 味 看 在 一 个 目 
录 以 及 它 的 所 有 子 目 录 中 禁用 per-directory 配 置 文件 。 程 序 清单 3.4 展 示 
了 了 如何 作为 一 个 整体 对 服务 器 禁用 per-directory 配 置 文件 。 这 提升 了 性 
能 ， 也 十 默认 的 Apache 配 置 。 


程序 清单 3.4 禁止 per-directory 配 置 文件 


1: <Directory /> 
2: AllowOverride none 
os. =/{DLrectory> 


3.7 Apache 日 志文 件 


默认 情况 下 ，Apache 包 含 两 个 日 志文 件 。access_log 文 件 用 于 跟踪 
客户 请 求 ;， error_log 文 件 用 于 记录 重要 事件 ， 例 如 错误 或 者 服务 郑重 新 
启动。 这些 文 件 从 你 第 一 次 局 动 Apache 时 束 和 存在 了， 这 些 文件 在 
Windows 平 侣 下 名 为 access.ljog 和 error.log。 


3.7.1 access_log X4} 


当 客 户 从 服务 器 请 求 一 个 文件 时 ，Apache 记 录 与 这 个 请 求 相 关 的 参 
数 ， 包 括 客户 的 IP 地 址 、 请 求 的 文档 、HTTP 状 态 码 和 当前 时 间 。 程 序 
清单 3.5 显 示 了 示例 日 志文 件 条 目 。 第 26 章 将 介绍 如 何 修改 已 记录 的 参 
数 。 
程序 清单 3.5 access_log 条 目 示 例 


1: 127.0.0.1 - - [12/Mar/202:08:33:25 -0700] "GET / HTTP/1.1° 200 44 
2: 127.0.0.1 - - [12/Mar/2012:08:33:25 -@70@] "GET /favicon.ico HTTP/1.1" 404 209 


3.7.2 error log 文件 


error log 文件 包含 错误 信息 、 局 动 信 息 和 服务 器 生命 周期 中 的 其 他 
重大 事件 。 当 你 使 用 Apache 遇 到 问题 时 ， 这 是 应 该 租 看 的 第 一 个 地 方 。 
程序 清单 3.6 展 示 了 error log 条目 示 例 。 


程序 清单 3.6 error log 条目 示 例 


: Starting the Apache2.4 service [The Apache2.4 service is running. ] 

: Apache/2.4.1 (Unix) configured -- resuming normal operations 

: [Tue Mar 13 @8:29:34 2012] [notice] Server built: Mar 12 2012 11:47:22 

: [Tue Mar 13 08:29:34 2012] [notice] Parent: Created child process 3504 
[Tue Mar 13 8:29:35 2012] [notice] Child 35@4: Child process is running 
[Tue Mar 13 08:29:35 2012] [notice] Child 3504: Acquired the start mutex. 


Oak wh — 


3.7.3 ”其 他 文件 


httpd.pid 文 件 包 侣 运行 Apache 服 务 需 的 进程 ID 号 。 可 以 手动 使 用 这 
个 号 但 发 送信 号 给 Apache， 接 下 来 的 一 节 将 详细 介绍 它 。scoreboard 文 
件 是 Linux/UNIX Apache 上 的 当前 配置 文件 ， 基 于 进程 的 MPM 用 这 个 文 
件 和 它们 的 子 进程 通信 。 一 般 来 说 ， 不 需要 考虑 这 些 文件 。 


3.8 ”Apache 相关 命令 

Apache 发 布 包 含 几 个 可 执行 程序 。 这 一 节 只 介绍 服务 器 二 进 制程 序 
和 相关 脚本 。 第 25 章 和 第 29 章 将 介绍 Apache 发 布 包含 的 其 他 应 用 程序 ， 
3.8.1 ”Apache 服务 左 二 进 制程 序 


Apache 可 执行 程序 在 Linux/UNIX 和 Mac OS X 中 叫做 httpd， 在 
Windows 中 叫做 apache.exe。 它 接受 几 个 合 令 行 选项 ， 参 见 表 3-1 中 的 说 
明 。 你 可 以 通过 在 Linux/UNIX 上 输入 /usr/local/apache2/bin/httpd -h , 
在 Mac OS X 上 输入 /usr/sbin/httpd -h ， 或 在 Windows 上 从 命令 行 提 示 输 
入 apache.exe -h ， 来 获取 一 个 完整 的 选项 列表 。 


表 3-1 ”httpd 选 项 


er 


列 出 编译 进 服务 砷 的 模块 


显示 版 本 写 和 服务 器 编 详 时 间 


如 果 和 编译 时 默认 值 不 同 ， 人 允许 传递 http.conf 的 位 置 





Apache 运 行 后 ， 可 以 在 Linux/UNIX 和 Mac OS X 上 使 用 kill 命 令 来 发 


AE Ss Za Apache KH E. fas hett— PRK ITS A EEL ERIK 
一 个 信号 ， 执 行 下 列 命 令 。 


# kill -SIGNAL pid 
在 这 个 语法 中 ，pid 是 进程 ID 号 ，SIGNAL 是 下 列 内 容 之 一 。 


TF IEARA ae 

e USR1 或 WINCH 一 一 温和 地 重新 局 动 ; 使 用 哪个 信号 取 雇 于 在 什 
么 操作 系统 下 。 

。 SIGHUP 一 一 重新 启动 。 





ode ed 们 立即 生效 ， 你 必须 及 信 
Se WA LEFF JB BY ARB as BY AIK — 
个 重新 局 动 的 信号 ， 你 可 以 做 到 这 一 点 。 这 告诉 Apache 重 新 谈 取 它 的 配 
ALIF. 


PEREISA EMAI AIBE ENS o AY BT A ARNA S 
种 不 同 的 方法 : 服务 于 一 个 客户 的 各 个 线程 或 进程 将 继续 执行 当前 请 
求 ， 但 当 它 结束 时 ， 它 将 被 终止 ， 并 用 一 个 采用 痢 配 置 的 新 线程 或 新 进 
BEANS Eo INE Webs at SILICA NFAL TE) HY FCAEER AE o 


在 Windows 上 ， 你 可 以 使 用 apache.exe 可 执行 程序 发 信号 给 
Apache。 


告诉 Apache 重 新 启动 。 
e apache.exe -k graceful 一 一 告诉 Apache 温 和 地 重新 启动 。 
告诉 Apache 人 停止 


e apache.exe -k restart 








e apache.exe -k stop 


你 可 以 在 开始 全 单项 中 访问 Apache 安 装 程序 创建 的 这 些 命令 的 快捷 
方式 。 如 果 把 Apache 作 为 一 项 服务 安 靖 ， 可 以 通过 使 用 Windows 服 务 界 
面 局 动 或 停止 Apache: 在 控制 面板 中 选择 管理 工具 并 点 击 服务 图 标 即 
可 。 


3.8.2 Apache 控制 脚本 


尽管 在 LinuxUNIX 上 使 用 httpd 二 进 制 代码 控制 Apache 是 可 能 的 ， 
但 是 推 基 使 用 apachectl 工 具 。apachectl 文 持 程序 把 通用 功能 包 思 到 一 个 
易于 使 用 的 脚本 中 。 要 使 用 apachectl， 输 入 如 下 命令 。 


# /usr/local/apache2/bin/apachectl command 
在 这 个 语法 中 ，command 可 以 是 stop start. restarteigraceful. KH 


可 以 编辑 apachectl 脚 本 的 内 容 以 讨 加 额外 的 命令 行 选 项 。 东 些 OS 友 布 提 
供 另 外 的 脚本 来 控制 Apache， 请 检查 发 布 中 包含 的 文档 。 


3.9 ”第 一 次 司 动 Apache 


在 局 动 Apache 之 前 ， 你 应 该 检查 最 基本 的 信息 设置 已 经 在 Apache 配 
置 文件 httpd.conf 中 完成 了 。 下 和 面 一 节 将 介绍 配置 Apache 并 局 动 服务 右 所 
需 的 基本 信息 。 


3.9.1 检查 你 的 配置 文件 


你 可 以 用 目 己 喜欢 的 文本 编辑 左 编 辑 Apache httpd.conf 文 件 。 在 
Linux/UNIX 和 Mac OS X 中 ， 它 可 能 是 vi 或 emacs。 在 Windows 中 ， 可 以 
用 记事 本 或 与 字 板 。 必 须 记 住 把 配置 文件 存 为 纯 文 本 格式 ， 这 是 Apache 
唯一 能 够 识别 的 格式 。 


第 一 次 司 动 Apache 只 有 两 个 参数 需要 修改 : MANA CURE 
监听 的 地 址 和 器 口 。 服 务 礁 的 名 字 是 当 Apache 需 要 引用 它 目 己 时 使 用 的 
一 个 参数 ,例如 重 定 同 请 求 的 时 候 。 


Apache 通 钊 可 以 从 机 老 的 卫 地 址 算出 它 的 服务 右 名 ， 但 这 不 是 绝对 

的 。 如 果 服 务 器 没有 一 个 有 效 的 DNS 条 目 ， 你 可 能 需要 指定 机 器 的 卫 地 
址 中 的 一 个 。 如 条 服务 硕 没 有 连接 到 网 络 〈 你 可 能 想 在 一 台独 立 的 机 项 
上 测试 Apache〉， 你 可 以 使 用 值 127.0.0.1， 这 是 一 个 回 送 地 址 

(loopback address) ,默认 病 口 写 是 80。 如 果 已 经 有 一 个 服务 器 运行 在 机 
右 的 80 冰 口上， 你 可 能 需要 改变 这 个 值 ; RA URRE Linux/UNIX A 
Mac OS X 系 统 上 没有 管理 员 权 限 ， 只 有 root 用 户 可 以 绑 定 到 体 留 端口 

(那些 小 于 1024 的 问 口 ) ， 那 可 能 也 需要 改变 这 个 端口 号 。 


你 可 以 用 Listen 指 令 改 变 监听 地 址 和 端口 值 。Listen 指 令 获 取 一 个 端 
口号 或 用 冒号 分 隔 开 的 一 个 IP 地 址 和 一 个 端口 。 如 果 只 指定 了 一 个 绒 
口 ，Apache 将 在 机 器 中 所 有 可 用 的 IP 地 址 上 监听 对 那个 端口 的 请 求 。 如 
果 还 提供 了 一 个 IP 地 址 ，Apache 将 只 监听 那个 地 址 和 痕 口 的 组 合 。 例 
ui, Listen 80 告 诉 Apache 在 所 有 卫 地 址 上 监听 对 80 问 口 的 请 求 。Listen 
10.0.0.1:443 47 YF Apache K thi WT 10.0.0.1_ E443 x9 H o 


ServerName S OVE URE LARS a8 4 WERKEN FIR 
给 所 有 目 引 用 的 URL。 这 个 指令 接受 一 个 DNS 名 字 和 一 个 可 选 的 奖 口 ， 
两 者 用 冒号 分 隔 开 。 确 定 ServerName 有 一 个 有 效 值 。 否 则 ， 服 务 器 将 不 
能 正 钊 运行 ， 例 如 产生 铬 误 的 重 定 问 。 


在 Linux/UNIX 和 Mac OS X 平 台 上 ， 你 可 以 使 用 User 和 Group 指令 来 
指定 服务 器 将 作为 哪个 用 户 和 组 ID 运行 。nobody 用 户 对 于 大 多 数 平 台 而 
音 是 一 个 好 的 选择 。 然 而 ， 在 HP-UX 平 台 用 这 个 用 户 ID 会 出 现 问题 ， 所 
以 必须 创建 并 使 用 一 个 不 同 的 用 户 ID， 例 如 www。 


3.9.2 ”局 动 Apache 


要 在 Linux/UNIX 上 启动 Apache， 切 换 到 包含 apachectl 脚 本 的 目录 并 
执行 下 面 的 命令 。 
# fusr/local/apache2/bin/apachectl start 

Mac OS X 用 户 可 以 在 命令 行 担 示 下 输入 如 下 命令 。 
# /usr/sbin/httpd 


要 在 Windows 上 手动 局 动 Apache， 在 开始 床单 中 的 Apache HTTP 
Server 2.2 程 序 组 中 ， 在 Control Apache Server 目 录 下 点 击 Start Apache in 


Console 链 搁 。 如 果 把 Apache 作 为 服务 安 故 ， 必 须 司 动 Apache 服 务 。 


如 有 果 一 切 顺 利 ， 你 可 以 使 用 浏览 器 访问 Apache。 显 示 一 个 默认 安 闭 
页 面 ， 如 图 3-4 所 示 。 如 果 不 能 局 动 Web 服 务 器 或 显示 一 个 错误 页 面 ， 
请 参阅 后 续 的 “故障 排除 ”一 三。 确定 你 在 Listen 指 令 中 指定 的 端口 之 一 
EY |=] Apache ——i85 #75 z 80 BK 808089 H o 





o=o | E x 
© localhost 
$ CO http://localhost I E 沁 
It works! 


3-4 E2 [ Apache 


3.10 ”故障 排除 


下 面 的 小 节 介 绍 了 第 一 次 启动 Apache 时 可 能 遇 到 的 几 个 常见 问题 。 
3.10.1 已 有 Web 服 务 器 


如 果 机 右上 已 经 运行 了 一 个 服务 右 并 监 昕 同样 的 IP 地 址 和 新 口 的 组 
合 ，Apache 将 不 能 成 功 局 动 。 你 将 在 错误 日 忘 文件 中 看 到 如 下 的 一 个 条 
目 ， 它 表明 Apache 不 能 绑 定 到 端口 。 


[crit] (48)Address already in use: make sock: could not bind...[alert] no 
listening sockets available, shutting down 


要 解决 这 个 问题 ， 需 要 停止 运行 的 服务 器 或 改变 Apache 配 置 以 监听 
一 个 不 同 的 端口 。 
3.10.2 不 允许 绑 定 到 端口 

如 果 没 有 管理 员 权 限 并 试图 绑 定 到 一 个 保留 病 口 (在 0 到 1024 之 
间 ) ， 你 将 得 到 一 个 如 下 的 错误 信息 。 


[crit] (13)Permission denied: make_sock: could not bind to address 10.0.8.2:80 
[alert] no listening sockets available, shutting down 


要 解决 这 个 问题 ， 就 必须 在 启动 Apache 之 前 作为 管理 员 登 录 或 改变 
端口 号 ，8080 是 一 个 经 常 使 用 的 非 保留 端口 。 


3.10.3 ”拒绝 访问 


如 果 没 有 权限 恋 取 配置 文件 或 写 入 日 忘 文 件 ， 你 可 能 不 能 局 动 
Apache 且 将 会 得 到 一 个 如 下 的 错误 信息 。 


(13}Permission denied: httpd: could not open error log file 


如 果 一 个 用 户 试 图 运行 Apache， 而 Apache 由 另 一 个 不 同 的 用 户 编 详 
并 安 闻 ， 将 可 能 产生 这 个 问题 。 


3.10.4 KAW 
你 可 以 配置 Apache 在 某 个 用 户 名 和 组 下 运行 。 对 于 运行 中 的 服务 器 
用 户 名 和 组 ，Apache 有 默认 值 。 有 时 默认 值 并 不 正确 ， 你 将 得 到 一 个 包 


A setgid: unable to set group id 的 错误 。 


HE Linux/UNIX H Mac OS X 上 解决 这 个 问题 ， 必 须 在 配置 文件 中 
把 Group 指令 的 值 修 改 为 一 个 正确 的 值 ， 为 已 有 的 组 检查 /etc/groups 文 
件 。 


2k Zs 


本 章 介 绍 了 在 LinuxUNIX、Mac OS X 和 Windows 上 安装 并 运行 
Apache 2.2 服 务 需 的 不 同方 法 。 其 中 包括 了 二 进 制 代码 和 产 人 代码 安 闭 ， 
并 介绍 了 基本 的 编 详 时 选项 。 另 外 ， 我 们 学 习 了 服务 器 配置 文件 的 位 置 
和 用 于 修改 Apache 配 置 的 命令 的 语法 。 我 们 学 习 了 两 种 主要 的 日 志文 
件 : access_log 和 和 error_log。 最 后 ， 我 们 学 习 了 在 Linux/UNIX、Mac OS 
X#llWindows-f & E tp faf 1E H Apache 4: t Hal A ak A pache Hk & as 2 ill FR 
AS J SA LEAR KE o 


3.12 Q&A 


Q: 我 如 何 能 够 月 动 一 个 清除 性 的 编 详 ? 


A: 如 果 你 需要 从 源 代 人 码 编译 一 个 新 Apache 并 且 不 想 早 期 编译 的 疆 
影响 新 的 编译 结果 ， 最 好 的 办 法 是 运行 make clean 命 令 。 它 将 仔细 清 
除 所 有 已 有 二 进 制 程序 、 中 间 对 象 文 件 等 。 


Q: per-directory 配 置 文件 有 什么 用 ? 


A: 尽管 per-directory 配 置 文件 对 服务 硕 性 能 有 影响 ， 但 它们 对 委托 
党 理 有 用 。 因 为 每 次 发 生 请 求 时 都 谈 取 per-directory 配 置 文件 ， 所 以 对 
配置 做 改变 时 不 需 重 新 司 动 服务 船 。 


你 可 以 允许 Web 站 点 的 用 户 不 必 拥有 管理 员 权限 就 能 自行 修改 配 
置 。 这 样 ， 他 们 可 以 分 部 分 地 对 他 们 的 主页 进行 密码 保护 。 


Q: 一 个 正确 的 ServerName 指 令 对 你 意味 着 什么 ? 


A: DNS 系 统 用 于 将 IP 地 址 和 域名 关联 起 来 。 当 服务 器 生成 一 个 
URL 时 人 返回 ServerName 的 值 。 如 果 要 使 用 某 一 域名 ， 你 必须 确定 它 包 合 
在 DNS 系统 中 并 且 访 问 站 点 的 客户 可 以 使 用 它 。 


3.13 KRAJ 


实践 练习 是 设计 用 来 帮助 你 预料 可 能 的 问题 、 复 习 已 经学 过 的 知 
识 ， 并 且 开 始 把 知识 用 于 实践 。 


Zk >) jel 
1. VRUN A RETRE Apache fhi B? 
2. <Location> $x Fi <Directory t Z EEK? 


3. Ha A A e E r a L AATA Ky? 


oy 


fa Z 
1. Linux/UNIX 用 户 可 以 使 用 configure 脚 本 的 --prefix 选 项 指定 安装 
Apache 的 位 置 。 如 果 在 这 个 位 置 已 经 有 一 个 安装 ， 配 置 文件 将 保留 ， 但 
是 二 进 制 文件 将 被 奉 换 。 在 Windows 中 ， 可 以 在 安装 向 导 中 设置 这 个 位 
置 。 


2. Directory 段 作用 于 文件 系统 对 象 ; Location 段 作用 于 Web 页 面 的 
地 址 栏 中 的 元 系 。 

3. 在 正 第 重新 司 动 期 间 ， 服 务 右 俘 止 ， 然 后 司 动 ， 这 将 导致 一 些 
请 求 丢 失 。 瘟 和 地 重新 局 动人 多 许 Apache 子 进程 /线程 继续 为 当前 的 请 求 
服务 ， 直 到 运行 新 配置 的 于 进程 /线程 能 够 取代 和 它们。 


BA He 
1， 练习 不 同类 型 的 服务 器 停止 运行 和 重新 启动 过 程 。 


2. 做 一 些 配置 修改 ， 例 如 不 同 痛 口 分 配 和 ServerName 变 化 。 


第 4 章 ”安装 和 配置 PHP 


在 本 章 中 ， 你 将 学 到 : 


。 Un] 2222 PHP. 

e 如 何 测 试 PHP 的 安装 。 

。 出 错 的 时 候 如 何 寻 求 帮助 。 
。 其 本 的 PHP 语 言 。 


在 本 草 中 ， 你 将 会 获取 、 安 友和 配置 PHP， 并 且 对 Apache 的 安 妆 做 
出 一 些 基本 的 改变 。 


4.1 PHP 的 当前 版 本 和 末 来 版 本 
本 章 中 的 安装 说 明 针 对 PHP 5.4.0， 这 是 该 软件 的 当前 版 本 。 


PHP Group 使 用 修订 和 小 版 本 的 发 布 来 更 新 所 包含 的 安全 性 扩展 或 
bug 的 修复 。 这 些 及 布 并 不 遵从 一 和 尽 及 布 时 间 规划 ， 当 扩展 或 修复 添加 
到 代码 中 并 经 过 彻底 的 测试 之 后 ，PHP Group 就 使 用 新 的 修订 号 发 布 一 
个 狐 的 版 本 。 


当 你 购买 本 书 的 时 候 ， 小 版 本 可 能 已 经 变化 到 5.4.1 或 更 局 版 本 。 如 
果 是 这 种 情况 ， 你 应 该 阅读 位 于 http://www.php.net/ChangeLog-5.php 的 
变化 列表 ， 米 了 解 天 于 安装 /配置 过 程 变 化 的 信息 。 这 些 过 程 是 本 章 的 
主要 内 容 。 


尺 官 在 两 个 次 版 本 更 新 之 间 不 可 能 所 有 的 安 北 过 程 部 要 变化 ， 但 你 
还 是 应 该 养 成 习惯 查看 自己 所 安装 和 维护 的 软件 的 更 新 日 志 。 如 有 果 在 你 
六 读本 书 的 时 候 ， 确 实 出 现 了 一 个 钦 版 本 的 变化 ， 但 更 新 日 志 中 并 没有 
所 到 安 冯 的 变化 ， 你 只 需要 用 心 记 下 ， 并 且 当 出 现在 安 儿 说 明和 相应 的 
图 中 的 时 候 ， 用 新 的 版 本 写 丛 代 束 行 了。 


4.2 在 带 有 Apache 的 LinuxwUNIX 上 编译 PHP 


m 本 节 中 ， 我 们 将 看 到 在 带 有 Apache 的 Linux/UNIX 上 安装 PHP 的 过 
时 。 这 个 过 程 对 于 任何 类 似 UNIX 的 操作 系统 来 说 多 少 都 有 些 相 同 。 尺 
“a 可 能 能 够 为 自己 的 系统 找到 PHP 的 预 编译 的 版 本 ， 但 是 从 源 代 码 编 
译 PHP 还 是 给 了 你 对 于 构建 到 上 自己 的 二 进 制 文件 中 的 功能 的 较 大 控制 


en 到 PHP 的 主页 http:V/www.php.net ， 并 且 找 到 
Downloads 部 分 。 找 到 了 最 新 版 本 的 源 人 代码， 例如 ， 我 们 使 用 的 是 5.4.0。 
你 的 发 布 pi 类 似 php-VERSION.tar.gz， 其 中 VERSION 是 最 近 的 
及 布 版 本 号 。 这 个 文档 将 会 是 一 个 压缩 的 tar 文 件 ， 因 此 ， 我 们 需要 解压 
Ae 
# gunzip < php-VERSION.tar.gz | tar xvf - 


把 下 载 文件 放 到 为 源 文件 保留 的 一 个 目录 中 ， 如 /usrsrc/ 
或 /usr/local/src/。 当 你 的 发 布 解压 后 ， 应 该 切换 到 PHP 发 布 目录 。 


# cd php-VERSION 


在 你 的 发 布 目录 下 ， 可 以 找到 一 个 名 为 configure 的 脚本 。 当 从 命令 
行 运行 configure 脚 本 的 时 候 ， 这 个 脚本 会 接受 所 提供 的 额外 信息 。 这 些 
命令 行 参数 控制 看 PHP 将 要 文 持 的 功能 。 在 这 个 例子 中 ， 我 们 将 会 包含 
在 Apache 和 和 MySQL 的 文 持 下 安装 PHP 所 需 的 基本 选项 。 稍 后 ， 我 们 还 将 
讨论 一 些 可 用 的 configure 选 项 ， 在 本 书 中 ， 它 们 都 是 相关 的 。 


# ./configure 9 --prefix=/usr/local/php \ 
--with-mysqli=/usr/local/mysql/bin/mysql config \ 
--With-apxs2=/usr/local/apache2/bin/apxs 


QO FR MS MySQLEV Apache 2228 FI y Six ANAC SPEED 
同 的 位 置 ， 那 么 ， 确 保 在 命令 中 用 相应 的 目录 路 径 进 行 奉 换 。 


脚本 运行 之 后 ， 会 返回 到 命令 提示 


Generating files 

updating cache ./canfig.cache 
creating ./config.status 

creating php. spec 

creating main/build-defs.h 
creating scripts/phpize 

creating scripts/manl/phpize. 1 
creating scripts/php-config 
creating scripts/mani/php-config. 1 
creating sapi/cli/php. 1 

creating main/php_config.h 
creating main/internal Tunctions.c 
creating main/internal_functions cli.c 


| License: | 
| This software is subject to the PHP License, available in this | 
| distribution in the file LICENSE. By continuing this installation | 
| process, you are bound by the terms of this license agreement. | 
| If you do not agree with the terms of this license, you must abort | 
| the installation process at this point. | 


Thank you for using PHP. 
# 


从 命令 提示 行 执 行 一 条 make 命 令 ， 接 着 执行 make install 命 令 。 这 些 
命令 将 会 完成 PHP 的 编译 和 安装 过 程 ， 并 且 返 回 命令 提示 行 。 


chmod 755 /fusr/local/apache2e/modules/libphp5. so 
[activating module ‘phps in /usr/local/apache2/conf /httpd. cont] 


Installing PHP CLI binary: fusr/local/php/bin/ 
Installing PHP CLI man page: fusr/local/php/php/man/mani / 
Installing PHP CGI binary: fusr/local/php/bin/ 
Installing build environment: fusr/local/php/lib/php/build/ 
Installing header files: fusr/local/php/include/php/ 
Installing helper programs: fusr/local/php/bin/ 


program: phpize 

program: php-config 
Installing man pages: fusr/local/php/ohp/man/mant ; 

page: phpize.1 

page: php-contig.1 
A 

你 还 需要 确 你 两 个 非常 重要 的 文件 复制 到 正确 的 位 置 。 自 完 ， 使 用 

如 下 命令 来 把 php.ini 的 推荐 版 本 复制 到 其 默认 位 置 。 稍 后 我 们 将 在 本 章 


中 学 习 有 关 php.ini 的 更 多 知识 。 
# cp php.ini-development /usr/local/1ib/php.ini 

接 下 来 ， 如 果 安 装 过 程 没有 把 PHP 共 享 对 象 文 件 复制 到 Apache 安 装 
目录 中 的 正确 位 置 (通常 安装 程序 会 复制 PHP 共 享 对 象 文 件 ， 从 make 
install 命 令 的 输出 可 以 看 到 ) ， 那 你 需要 执行 如 下 命令 。 
# cp libs/libphp5.so /usr/local/apache2/modules/ 

你 现在 应 该 可 以 配置 和 运行 Apache 了 ， 但 是 ， 在 直接 开始 “在 
Linux/UNIX 上 整合 PHP 和 Apache” 这 一 节 之 前 ， 让 我 们 先 介 绍 一 些 额外 
的 配置 选项 。 


4.2.1 额外 的 Linux/UNIX 配 置 选 项 


在 前 面 的 一 节 中 ， 当 运行 PHP configure 脚 本 的 时 候 ， 我 们 可 以 包含 
一 些 命 令 行 参 数 来 确定 PHP 引 擎 将 要 包含 的 那些 功能 。configure 脚 本 本 


导 给 出 了 一 个 可 用 选项 的 列表 ， 包 括 我 们 所 使 用 过 的 那个 列表 。 从 PHP 
发 布 的 目录 ， 输 入 如 下 命令 。 
# ./configure --help 

这 条 命令 产生 一 个 长 长 的 列表 ， 以 便 你 将 其 存 入 到 文件 并 且 在 空 有 
的 时 候 阅 读 它 。 
# ./configure --help > configoptions.txt 


ON RTE PHP ZRI Ja WIAA Uh A Be A SI BIPHP PA i 
要 再 次 运行 配置 和 编译 过 程 。 这 么 做 将 会 生成 一 个 狐 版 本 的 libphp5.so， 
并 且 将 其 放置 到 Apache 的 目录 结构 中 。 你 所 需要 做 的 只 是 重新 司 动 
Apache 以 载 入 新 文件 。 


4.2.2 ”在 LinuxwUNIX 上 集成 PHP 和 Apache 


要 确 你 PHP 和 Apache 能 够 协同 工作 ， 我 们 需要 检查 httpd.conf 配 置 文 
件 并 且 潜 在 地 同 其 中 添加 一 些 项 目 。 育 先 ， 看 看 如 下 的 一 行内 容 。 
LoadModule php5 module modules/1ibphp5. so 

如 果 这 行内 容 没有 出 现 ， 或 者 在 这 行 的 开头 出 现 了 一 个 “天 号 ， 你 
必须 诬 加 一 行 或 者 删除 这 个 “ 朝 号 。 这 一 行 竺 诉 Apache 使 用 PHP 编 译 过 
程 所 创建 的 PHP 共 享 对 象 文件 dibphp5.so)。 


Be POR, AERO PAA. 


# 

# AddType allows you to add to or override the MIME configuration 
# Tile mime.types for specific file types. 

# 


FLIX — ADOT A AS SAF íT o 
AddType application/x-httpd-php .php 


这 条 语句 确保 了 PHP 引 擎 将 会 解析 以 .php 和 .html 扩 展 名 结尾 的 文 
件 。 你 所 选择 的 文件 名 可 能 有 所 不 同 。 


体 存 这 个 文件 ， 然 后 重新 局 动 Apache。 当 你 三 看 目 己 的 error log 的 
时 候 ， 应 该 会 看 到 如 下 的 一 行 


[Tue Mar 13 10:42:47 2012] [notice] Apache/2.4.1 (Unixy PHP/5.4.0 configured 


PHP 现 在 已 经 是 Apache ia ASHI ebay S o URIA T feU 
何在 Mac OS X 平 台 上 安装 PHP， 请 继续 了 阅读。 否则 ， 可 以 跳 到 “测试 安 


J ?? e 


4.3 在 MacOSX 上 安装 PHP 


在 带 有 Apache 的 Mac OS X 上 安装 PHP 有 几 个 不 同 的 选项 ， 包 括 前 面 
小 节 所 介绍 的 从 源 代码 编译 。 一 些 用 户 可 能 会 发 现 ， 最 简 蛙 的 方法 是 从 
一 个 预 编 详 的 二 进 制 包 来 安装 PHP， 例 如 ，MacPorts 
(在 http:/www.macports.org/ )， 它 是 一 次 性 安 痛 包 XAMPP 的 一 部 分 ， 或 
者 MAMP (在 http://www.mamp.info )。 然 而 ， 如 果 你 习惯 使 用 命令 行 ， 
我 建议 你 按照 4.2 节 的 说 明 来 进行 。 


4.4 在 Windows 上 安装 PHP 


人 在 Windows 上 和 安 泪 PHP 也 只 需要 下 载 友 布 文件 而 已 。 要 下 载 PHP 友 
布 文 件 ， 请 到 PHP 的 主页 http://www.php.net/ ， 按 照 链接 找到 Downloads 
页 面 。 找 到 线程 安全 的 ZIP 包 的 最 独 版 本 ， 例 如 ， 我 们 使 用 的 是 5.4.0。 
你 下 载 的 发 布 的 名 字 将 会 类 似 于 php-VERSION.zip， 其 中 VERSION 是 最 
UA AA RASS o 
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径 名 已 经 他 在 了 ， 因 此 ， 把 文件 解压 缩 到 名 为 C\php\ 的 一 个 新 目录 ， 
所 有 的 文件 和 子 目录 都 将 放置 到 这 个 新 目 孙 下 面 。 


接 看 ， 访 问 Ci\php\ 目 录 并 把 文件 php.ini-recommended 复 制 为 
php.ini。 为 了 获得 和 Apache 一 起 使 用 的 PHP 的 基本 版 本 ， 我 们 需要 对 
Apache 配 置 文件 做 一 些 细微 的 修改 。 


ae 
YES: 


在 一 些 Windows 系 统 上 ， 你 可 能 需要 设置 一 个 明确 的 环境 变量 ， 以 便 让 PHP 正 确 地 运行 ; 
即便 你 不 确定 这 个 变量 是 侣 是 必须 的 ， 设 置 它 也 不 会 引起 什么 危害 ， 因 此 ， 没 有 理由 不 这 人 么 
做 。 要 了 解 将 PHP 目 录 添 加 到 PATH 环境 变量 的 更 多 信息 ， 参 见 位 于 
http://www. php.net/manual/en/faq.installation.php#faq.installation.addtopath 的 PHP FAQ 的 相关 条 
Ae 





TE Windows E ® PHPF Apache 


BT PRPHP H Apache t HE CIE, RI] TX httpd.conf E. x4 


添加 一 些 项 日 。 首 先 ， 找 到 httpd.conf 配 置 文件 如 下 的 部 分 。 
# Example: 

# LoadModule foo_module modules/mod foo.so 

# 

LoadModule access module modules/mod access.so 


#LoadModule vhost alias module modules/mod vhost alias.so 


FEET HARE, USING PA 


LoadModule php5 module C:/php/phpS5apache2 2.d11 


Sook, WAN BAY AY A WA a RA pache #18 php.ini hA E 6 
PHPIniDir "G:/php/" 

接 下 来 ， 找 到 如 下 的 部 分 。 
# 
# AddType allows you to add to or override the MIME configuration 


# file mime.types for specific file types. 
# 


EXTR EEA BA íT o 
AddType application/x-nttpd-php .php 
这 条 语句 硝 保 了 PHP 引 擎 可 以 解析 以 .php 和 .htm] 朱 展 名 为 结尾 的 文 
件 。 你 对 文件 名 的 选择 可 能 有 所 不 同 。 


保存 httpd.conf 文 件 ， 然 后 重新 司 动 Apache。 服 务 右 局 动 后 应 该 没有 
警告 ，PHP 现 在 已 经 是 Apache Web 服 务 需 的 一 部 分 了 。 


4.5 php.inizétit 


当 你 编译 或 安 闻 了 了 PHP 之 后 ， 仍 然 可 以 使 用 php.ini 文 件 来 改变 其 行 
为 。 在 LinuxXUNIX 系 统 上 ， 这 个 文件 的 默认 位 置 是 /usvlocalphpy/lib， 或 
者 是 在 配置 时 所 使 用 的 PHP 安 装 位 置 的 lib 子 目录 。 在 Windows 系 统 中 ， 
这 个 文件 应 该 在 PHP 目 录 中 ， 或 者 在 Apache httpd.conf 文 件 中 的 
PHPIniDir 值 所 指定 的 另外 一 个 目录 中 。 


php.ini 文 件 中 的 指令 有 两 种 格式 : 仁和 标记 。 值 指令 的 格式 是 一 个 
指令 名 以 及 一 个 等 号 隅 开 的 一 个 值 。 可 能 的 值 对 于 不 同 的 指令 来 说 各 有 
不 同 。 标 记 指 令 的 格式 是 一 个 指令 名 以 及 一 个 等 号 陋 开 的 一 个 正 的 或 负 
的 项 。 正 的 项 包括 1、On、Yes 和 True; 负 的 项 包括 0、Off、No 和 
False; 2A Alig. 


H 
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在 Windows 系 统 上 ， 和 重要 的 是 为 extension_dir 指 令 提 供 正 确 的 值 。 如 果 你 将 PHP 安 装 在 
Ci:\php 位 置 ， 那 么 ，extension_dir 的 值 应 该 是 “C:\php\ext”。 





你 可 以 随时 改变 php.ini 的 设置 ， 但 是 改变 之 后 ， 圾 要 章 新 局 动 服务 
名 以 便 让 改变 生效 。 在 这 里 ， 不 妨 伦 扣 时 间 阅 读 一 下 目 己 的 php.ini 文 
件 ， 看 看 可 以 配置 的 项 目 有 哪些 。 


4.6 ”测试 安装 


测试 PHP 安 竣 的 最 简单 的 方法 融 是 使 用 phpinfo0 函 数 创建 一 个 小 的 
汕 弃 脚本 。 这 个 函数 将 会 产生 一 个 长 长 的 配置 信息 列表 。 打 开 文 本 编辑 
厂 并 输入 如 下 的 命令 行 。 


<?php phpinfo(); ?> 


把 这 个 文件 保存 为 phpinfo.php， 并 且 将 其 放置 到 Web 服 务 占 的 文档 
根 目 录 下 ， 即 Apache 安 闭 的 htdocs 子 目录 或 者 Mac OS XE 
的 /Library/WebServerDocuments。 使 用 Web 浏 览 右 访问 该 文件 ， 你 应 该 
会 看 到 如 图 4-1 所 示 的 内 容 。 


/图 phpinfa() 
© C | © http://localhost/phpinfo.php 


= 





Build Date | Feb 29 2012 19:20:16 
MSVC9 (Visual C++ 2008) 


Configure | cscript /nologo configure.js "—enable-snapshot-build” “—disableisapi" "—enable-debug-pack” 
Command “_-disable-nsapl” “—withoutmssal” “—without-pdo-mssql* “—without-pisweb° “—with-pdo- 
oci=G:\php-sdkioraclelinstantclientt Oisdk shared” “—with-ocié=C:iphp- 
sdkoracle\instantclienti O\sdk, shared “—with-oci8-1 19=C:\php- 
| sdkioraclelinstantclientl lisdk, shared" “—-enable-objectout-dir=_/obj" “—-enable-com-dotnet 
| *--with-mcrypt=static® "—disable-static-analyze" “—with-pgo” 


apache 20 Mande 


Scan this dir | (none) 
for additional 

.ini files 

Additional .ini | (none) 
as 


PHP API ‘| 20100412 


20100525 





a aih m -一 -一 二 -一 -一 


Iir | 



























































图 4-1 ” phpinfo() 的 结 





phpinfo() 具 体 的 输出 取决 于 你 的 操作 系统 、PHP 版 本 以 及 配置 选 


项 。 


4.7 FRA AEA 


通过 Internet 获 取 帮 助 总 是 很 方便 ， 尤 其 是 对 于 涉及 到 开源 软件 的 问 
ml. YAM, TEPER ARISTA ZA, AOSTA. WEI ER. BE 
或 编程 问题 看 上 去 如 何 难 以 解决 ， 但 肯定 不 会 只 有 你 才 过 到 这 种 情况 
可 能 有 人 已 经 解决 了 你 的 问题 。 


当 我 们 过 a 到 困难 ， 自 和 完 应 该 求助 的 资源 束 古 PHP 官 方 网 站 
eee php.net/ 《尤其 是 使 用 说 明 手 册 http://www.php.net/manual/ 

。 如 果 在 这 里 没有 找到 答案 ， 列 喜 了 PHP 站 点 是 可 以 搜索 的 。 你 所 寻 
aed 一 个 友 布 消 居 中 或 者 在 一 个 音 见 问题 解答 文 
件 中 。 你 也 可 以 在 http://www.php.net/search.php 搜索 邮件 列表 文档 。 

些 文档 凝 办 了 了 PHP 社区 的 很 多 乔 意 ， 并 且 提 供 了 非常 多 的 信息 资源 。 不 
妨 化 点 时 间 答 试 几 个 搜索 关键 词 的 组 合 。 


如 末 确 定 目 己 的 问题 没有 被 解决 过 ， 可 以 同 PHP 社 区 提出 问题 并 得 
到 服务 。 你 可 以 从 http:/www.php.net/mailing-lists.php 加 入 到 PHP 邮 件 列 
Ko RE Ri mA ARAN RAS, (Ape Male 
多 。 如 采 你 专门 从 事 PHP 脚 本 编程 ， 至 少 应 该 订阅 一 个 摘要 列表 ， 当 你 
订阅 了 和 目 己 相关 的 列表 ， 可 以 考 处 发布 你 的 问题 。 


当 你 友 布 一 个 问题 的 时 候 ， 包 含 尽 可 能 多 的 信息 是 个 不 错 的 想法 ， 
(AS BUR UALR rs FARE UN AY Tis AGE P IRIE E o 


。 你 的 操作 系统 类 型 。 
。 你 所 运行 或 安装 的 PHP 版 本 。 


。 你 的 配置 选项 。 
© 在 安装 失败 之 前 的 configure 或 make 命 令 的 任何 输出 。 
。 | 发 问题 的 代码 的 一 个 相对 完整 的 示例 。 


为 什么 所 有 这 些 都 和 在 一 个 邮件 列表 张贴 问题 有 头 呢 ? A 
研究 技能 会 对 你 大 有 人 宰 蔡 。 一 个 好 的 研究 者 通 第 能 够 快速 而 局 效 地 解决 
问题 。 在 一 个 技术 列表 中 张贴 一 个 幼稚 的 问题 ， 往 往 会 需要 等 符 而 得 到 
的 只 是 一 条 消息 或 者 两 个 引用 而 已 ， 它 们 只 能 够 让 你 知道 应 该 首先 从 哪 
里 去 伍 找 问题 的 舍 采 。 


其 次 ， 别 二 了 一 个 邮件 列表 并 不 像 一 个 技术 文 持 呼叫 中 心 。 没 有 人 
会 因为 回答 你 的 问题 而 得 到 报酬 。 尽 管 如 此 ， 你 还 是 会 看 到 一 个 令 人 难 
环 的 镶 意 和 知识 的 至 库 ， 其 中 包括 一 些 PHP 的 创造 者 的 思想 。 一 个 好 的 
问题 及 其 解答 将 会 被 存档 ， 以 帮助 其 他 的 程序 员 。 多 次 询问 已 经 解答 的 


问题 则 只 会 增加 更 多 的 噪音 。 


尽管 如 此 ， 不 要 担心 把 问题 张贴 到 邮件 列表 中 。PHP 开 及 者 是 一 群 
文明 而 乐于 助人 的 人 ， 而 且 通 过 让 问题 引起 社区 的 注意， 你 可 能 也 帮助 
其 他 人 解决 了 同样 的 问题 。 


4.8 PHP 脚 本 基础 


让 我 们 直接 跳 到 PHP 脚 本 。 首 先 ， 打 开 你 所 喜爱 的 文本 编辑 器 ， 像 
HTML 文 档 一 样 ，PHP 文 件 也 是 由 纯 文本 构成 的 。 你 可 以 使 用 任何 文本 
编辑 需 来 创建 它们 ， 并 且 ， 大 多 数 流行 的 HTML 编辑 器 和 编程 IDE〈 集 
成 开发 环境 ) 都 支持 PHP。 


寻找 PHP 友 好 的 编辑 共有 的 一 个 不 错 的 站 点 是 www.php-editors.com。 
输入 程序 清单 4.1 中 的 例子 并 将 文件 保存 到 Web 服 务 问 的 根 目 录 ， 将 其 命 
名 为 first.php。 


程序 清单 4.1 一 个 人 简单 的 PHP 脚 本 


1: <?php 
2: echo "<h1>Hello Web!</h1>"; 
3: ?> 


如 果 不 能 在 为 PHP 脚 本 提供 运行 服务 的 机 器 上 直接 工作 ， 则 可 能 需 
要 一 个 FIP 或 SCP 客 户 问 把 保存 的 文档 上 传 到 服务 器 。 当 文档 位 于 服务 
器 上 的 适当 位 置 时 ， 你 就 可 以 使 用 浏览 器 来 访问 它 了 。 如 果 一 切 正 党 ， 
将 会 看 到 脚本 的 输出 。 图 4-2 显 示 了 脚本 first.php 的 输出 。 


() localhost/first.php 
€ C | © http://localhost/firs 


Hello Web! 





图 4-2 ”成 功 :脚本 firstphp 的 输出 
4.8.1 开始 和 结束 一 个 PHP 语 句 块 


在 编写 PHP 的 时 候 ， 你 需要 通知 PHP 引 擎 你 想 执行 的 是 哪些 命令 。 
如 末 不 能 通知 PHP 引 擎 这 些 命 令 ， 所 编写 的 代码 将 会 被 误 认 为 HIML 并 
且 直 接 和 输出 到 浏览 器 。 你 可 以 把 上 自己 的 代码 设计 为 市 有 专门 标记 的 
PHP， 这 些 标记 表示 了 PHP 代 码 块 的 开始 和 结束 。 表 4-1 给 出 了 4 种 PHP 
分 隅 标记 。 


PHP 开 始 和 结束 标记 





脚本 标记 <script language="php"> </script> 





在 表 4-1 所 示 的 标记 中 ， 只 有 标准 标记 和 脚本 标记 保证 对 任何 配置 
都 有 效 。 短 标记 和 ASP 式 标记 必须 在 php.ini 中 显 式 地 启用 。 


要 激活 对 短 标 记 的 识别 ， 必 须 确保 在 php.ini 中 将 short _open_tag 设 置 
为 On。 
sho rt open tag = On p 
要 激活 对 ASP 却 标记 的 识别 ， 必 须 确 保 在 php.ini 中 将 asp_tags 设 置 为 
On. 
asp tags = On; 
提示 : 


要 确保 可 移植 的 、 可 复 用 的 代码 ， 最 好 使 用 标准 标记 而 不 古 短 标记 或 ASP 式 标记 。 原 因 很 
人 简单， 服务 画 的 配置 总 是 唯一 的 ， 使 用 标准 标记 是 因为 你 知道 在 任何 配置 上 都 可 以 使 用 它 。 





在 编辑 了 php.ini 并 且 重 新 启动 Apache 之 后 ， 就 可 以 在 脚本 中 使 用 四 
种 标记 类型 中 的 任何 一 种 了 。 这 在 很 大 程度 上 与 个 人 偶 好 有 天， 当然 如 
果 你 有 意 要 在 自己 的 脚本 中 包含 XML， 你 应 该 避免 使 用 短 标 记 (<? ?>) 而 
使 用 标准 标记 (<?php ?>)。 


Soe SS 
VERS: 





字符 序列 <? 告 诉 一 个 XML 解析 器 将 有 一 个 处 理 指令 ， 因 此 该 序列 经 常 包 含 在 XML 文档 
中 。 如 果 你 在 脚本 中 包含 XML 并 且 短 标记 可 用 ，PHP 引 警 束 可 能 混 消 XML 处 理 指令 和 PHP 开 
始 标记 。 如 果 你 有 意 在 文档 中 加 入 XML， 那 么 就 关闭 短 标 记 。 








让 我 们 给 出 程序 清单 4.1 中 的 代码 的 一 些 合法 写法 。 你 可 以 使 用 已 
经 见 到 的 4 个 PHP 开 始 和 结束 标记 中 的 任何 一 个 。 


<? 
echo "Hello Web!"; 
?> 


<?php 
echo "Hello Web!": 
二 


<% 


echo "Hello Web!"; 


o> 
<script language=" php > 


echo "Hello Web!"; 
</script> 


你 也 可 以 把 单行 代码 放 入 到 与 PHP 开 始 和 结束 标记 相同 的 一 行 中 ， 
示例 如 下 。 
<?php echo "Hello Web!"; 7> 


现在 ， 你 知道 如 何 定 义 一 段 PHP 人 代码， 下 面 让 我 们 进一步 看 看 程序 
is 241A GING XM. 


4.8.2 echoif f) print() 2K 2% 


人 简单 地 说 ，echo 语 句 用 来 输出 数据 。 在 大 多 数 情 况 下 ，echo 的 任何 
和 输出 最 终 在 浏 虎 硕 中 都 是 可 见 的 ， 也 可 以 使 用 printO 国 数 来 答 代 echo 语 
伍 ]。 使 用 echo 或 printO 只 是 个 人 习惯 的 问题 ， 当 你 查看 其 他 人 的 脚本 
时 ， 两 种 用 法 都 会 见 到 。 


再来 看 看 到 目前 为 止 我 们 所 见 到 过 的 代码 ， 注 意 程 序 清 单 4-1 中 以 


分 写 结束 的 唯一 那 行 代码 。 这 个 分 号 告诉 PHP 引 擎 ， 一 条 语句 结束 了 了 ， 
并 且 ， 这 十 目 前 为 止 我 们 所 学 到 的 关于 编码 语法 的 最 午 要 的 一 反 。 


WAJ (statement) 表示 对 PHP 引 擎 的 一 条 指令 。 更 广泛 地 讲 ， 对 于 
PHP 来 说 ， 它 了 驶 像 是 用 丙 语 写 或 说 的 一 个 名 于 。 一 个 英语 句子 通 名 应 充 
用 一 个 句点 结束 ， 一 条 PHP 语 句 通 第 应 该 以 一 个 分 号 结束 。 这 条 规则 的 
例外 之 一 是 包含 了 其 他 语句 的 语句 ， 以 及 结束 一 个 代码 块 的 语句 。 然 
而 ， 在 大 多 数 情 况 下 ， 没 有 用 分 号 结束 一 条 语句 将 会 引起 PHP 引 擎 的 泥 
清 并 且 产 生 一 个 错误 。 


4.8.3 组 合 HTML 和 PHP 


程序 清单 4.1 中 的 脚本 是 纯 PHP 的 。 只 需要 简单 地 把 HTML 加 到 PHP 
开始 和 结束 标记 的 外 围 ， 就 能 够 把 这 个 脚本 加 入 到 一 个 HTML 文档 中 ， 
如 程序 清单 4.2 所 示 。 


程序 清单 4.2 ”加 入 到 HTML 中 的 一 个 PHP 脚 本 


: <!IDOCTYPE html> 

: <html> 

: <head> 

: <title>A PHP script including HTML</title> 
</head> 

<body> 

: <h1><?php echo "hello world"; ?></h1> 

: </body> 

: </html> 


oo 上 co hm 一 


正如 你 所 见 到 的 ， 把 PHP 代 码 加 入 到 一 个 主 挖 HTML 文 档 中 只 是 代 
码 输 入 上 的 小 问题 。PHP 引 擎 忽略 PHP 开 始 和 结束 标记 外 围 的 所 有 内 
容 。 如 果 你 把 程序 清单 4-2 中 的 内 容 保 存 为 helloworld.php， 并 将 其 放置 
到 文档 根 目 录 ， 然 后 使 用 浏览 占 查 看 它 ， 如 图 4-3 所 示 ， 你 可 以 看 到 粗 


体 的 hello world。 如 果 你 要 答 看 文档 源 文 件 ， 如 图 4-4 所 示 ， 清 单 看 上 去 
确实 像 一 个 普通 的 HTML 文 档 。 


(©) A PHP script including HT! 


e C © http://localhost/helloworld.php 


hello world 





图 4-3 helloworld.php 的 输出 在 浏览 器 中 的 样子 





ls view-source:localhost/ hell: 


@ | @ view-source:localhost/helloworld.ohp sr Kk 


2 | <html> 

3| <head> 

4|/<title>A PHP script including HIML</title> 
5 | </head> 

3| <body> 

T|<hi>hello world</hi> 

B| </body> 

3| </html> 


图 4-4 ”helloworld.php 的 输出 的 HTML 源 代码 


只 要 你 愿意 ， 你 可 以 在 一 个 单个 的 文档 中 包含 任意 多 个 PHP 代 码 
块 ， 这 些 代 人 码 块 可 以 根据 需要 散布 在 HTML 中 。 尺 管 可 以 在 一 个 单个 的 
文档 中 有 多 个 代码 块 ， 但 可 以 将 它们 看 做 一 个 单个 的 脚本 。 定 义 在 第 一 


A SERB Ee ee, A tA DA PE Se A TRS RAE H o 
4.8.4 ”为 PHP 代 码 添 加 注释 


那些 在 编写 的 时 候 看 上 去 干 干 判 穴 的 代码 ， 当 你 6 个 月 后 试图 改进 
它 的 时 候 ， 看 上 去 是 那么 的 杂乱 无 草 。 在 编 与 代 但 的 时 候 为 代码 添加 注 
释 ， 以 后 将 会 节省 你 的 时 间 ， 并 且 使 得 其 他 的 程序 员 更 容易 使 用 你 的 代 
码 。 


注释 Ccomment) Wize IAS 4kPHP S| AMR SCA. TERER LA 
使 得 代码 更 具 可 读 性 ， 或 者 用 来 说 明 一 个 脚本 。 


单行 注释 以 两 个 斜 杜 开始 (V//)， 这 是 首选 的 方式 ， 也 可 以 以 一 个 斜 
杠 或 一 个 井 号 开始 。PHP 引 擎 忽略 这 些 符 号 到 行 末 或 者 这 些 符 号 到 PHP 
结束 标记 之 间 的 所 有 文本 。 


// this is a comment 
# this is another comment 


BATTER “as tL a ERA PES (ITER, ARRES 
OS ES Ja LER a ANRH « 


/* 

this is a comment 
none of this will 
be parsed by the 

PHP engine 

ae 


49 小结 


在 本 章 中 ， 我 们 学 习 了 如 何在 Linux/UNIX、Mac OS X 或 Windows 
上 安装 和 配 普 PHP 5.4.0， 以 便 和 Apache 一 起 使 用 。 学 习 了 在 
Linux/UNIX 编 译 脚 本 中 的 各 种 configure 选 项 ， 可 以 修改 它们 来 改变 所 文 
持 的 功能 。 还 学 习 了 有 关 php.ini 的 知识 ， 以 及 如 何 改变 其 指令 的 值 。 


使 用 phpinfoO 函 数 ， 我 们 可 以 测试 安装 并 且 产 生 其 配置 值 的 一 个 列 
表 。 我 们 使 用 一 个 文本 编辑 髓 创建 了 一 个 简单 的 PHP 脚 本 。 我 们 介绍 了 
可 以 用 来 开始 和 结束 PHP 代 码 块 的 4 种 标记 。 


WA Aa 


最 后 ， 我 们 学 习 了 如 何 使 用 echo 语 句 或 printO 函 数 癌 浏览 器 发 送 数 
据 ， 并 且 把 HIML 和 PHP 脚 本 组 合 到 同一 个 文件 中 。 在 下 一 章 ， 我 们 将 
使 用 这 些 技巧 来 测试 PHP 语 言 的 一 些 基本 组 成 部 分 ， 包 括 变 量 、 数 据 类 
型 和 操作 符 。 


4.10 Q&A 


Q: 我 们 已 经 介绍 了 LinuxUNIX、Mac OS X, Windows I X 
Apache Web 服 务 硕 的 PHP 安 猴 。 这 征 售 意味 看 本 书 介绍 的 内 
容 不 适用 于 我 的 服务 右 和 操作 系统 ? 


A: 不 是 的 。PHP 的 一 个 最 大 优点 束 是 它 可 以 在 多 种 平台 上 运行 。 
你 可 以 在 PHP 手 册 中 找到 针对 不 同 Web 服 务 器 和 数据 库 文 持 的 不 同 配置 
指令 的 安装 说 明 。 尺 管 本 书 中 的 例子 都 是 特别 针对 PHP、MySQL 和 
Apache 的 ， 但 在 使 用 不 同 的 web 服务 需 或 数据 库 的 时 候 ， 只 需要 略 作 修 
改 束 可 以 使 用 这 些 例子 。 


Q: 最 好 的 开始 和 结束 标记 是 什么 ? 


A: 这 在 很 大 程度 上 是 个 人 偏好 的 问题 。 但 为 了 可 移植 性 ， 标 准 标 
记 (<?php ?>) 是 最 佳 的 选择 。 


Q: 在 编写 PHP 代 码 的 时 候 ， 应 该 避免 使 用 哪些 编辑 器 ? 
A: 不 要 使 用 为 了 打印 而 格式 化 文本 的 字 处 理 软件 〈 如 Microsoft 
Word) 。 即 便 使 用 这 种 软件 以 纯 文 本 的 格式 保存 你 所 创建 的 文件 ， 隐 


IRRI FITE n Re E o 


Q: FATE ATE PER ANS ? 


A: 这 又 是 一 个 个 人 偏好 的 问题 。 某 些 较 短 的 脚本 具有 很 好 的 自 解 
释 性 ， 即 便 在 很 长 一 段 时 间 之 后 也 是 如 此 。 不 过 脚本 的 长 度 和 复 洒 性 如 
何 ， 痢 应 该 注释 你 的 代码 。 从 长 远 来 讲 ， 注 释 代 人 码 第 常会 节省 你 的 时 
E, IRD HEITI. 


实践 练习 


这 个 实践 练习 设计 用 来 帮助 你 预测 可 能 的 问题 、 复 习 已 经 学 过 的 知 
识 ， 并 且 开 始 把 知识 用 于 实践 。 


问答 题 
1. 从 Linux/UNIX 操 作 系 统 ， 你 如 何 获 得 有 关 配 置 选项 (你 传递 给 
PHP 发 布 中 的 configure 脚 本 的 选项 ) 的 帮助 ? 


2. 为 了 人 确保 能 够 识别 .php 扩 展 名 的 文件 ， 你 应 该 在 Apache 配 置 文 
件 中 添加 一 行 什么 ? 


3. PHP 的 配置 文件 叫 什么 ? 


4. 一 个 人 浏览 你 的 Web 站 点 可 以 读 取 已 经 成 功 安 装 的 PHP 脚 本 的 
WAAR AG ? 


oy 


ee 


1. 可 以 通过 在 PHP 发 布 目 录 下 调用 configure 脚 本 并 把 --help 参 数 传 
BZ’ 从 而 获取 有 天 配 置 选项 的 帮助 信息 Jay 9 命 x 令 如 下 。 


./configure --help 


2. 确 你 Apache 能 够 把 .php 扩 展 名 的 文件 当 作 PHP 脚 本 的 一 行 命令 如 
Fe 


AddType application/x-httpd-php .php 


3. PHP 的 配置 文件 叫做 php.ini。 


4. 不 能 ， 这 个 用 尸 只 能 看 到 脚本 的 输出 。 


FAG el 


1. EMMARERAPHP. WRCR 了， 但 看 php.ini 文 件 并 检 
碍 你 的 配置 。 


2 熟悉 创建 、 上 传 和 运行 PHP 脚 本 的 过 程 。 特 别 是 创建 你 目 己 
的 “hello world” 脚 本 。 为 它 生 加 HIML 人 代码， 并 且 添 加 其 他 的 PHP 语 名 
块 。 使 用 不 同 的 PHP 分 隔 符 标记 来 实验 。 你 的 配置 文 持 哪 一 个 标记 ? 看 
一 下 php.ini 文 件 以 证 实 你 的 发 现 。 别 扎 了 为 你 的 代码 添加 一 些 注 释 。 


第 2 部 分 “PHP 语言 结构 


PHP 的 组 成 部 分 
PHP 的 流程 控制 功能 
使 用 函数 

使 用 数组 


使 用 对 象 


第 5 音 “PHP 的 组 成 部 分 


在 本 章 中 ， 你 将 学 到 : 


变量 一 一 它们 十 什么 、 为 什么 我 们 需要 使 用 它们 ， 以 及 如 何 使 用 
它们 。 

如 何 定义 和 访问 变量 。 

数据 类 型 。 

一 些 很 第 用 的 操作 符 。 

如 何 使 用 操作 和 从 来 编写 表达 式 。 

an TEJ FE X AITE H E E o 


在 本 章 中 ， 我 们 将 亲身 了 解 一 些 PHP 脚 本 语言 的 具体 细节 。 新 接触 
编程 的 读者 可 能 会 觉得 不 知 所 措 ， 不 过 不 用 担心 ， 你 以 后 可 以 随时 回 过 
头 来 参考 本 章 。 请 把 注意 力 放 在 对 概念 的 理解 上 ， 而 不 是 记 住 所 介绍 的 
功能 上 ， 因 为 这 些 元 系 会 在 本 书 的 脚本 中 重复 出 现 。 如 末 第 一 次 没有 党 
握 它 们 ， 你 最 终 还 是 会 掌握 它们 的 。 


如 果 你 已 经 是 一 个 有 经 验 的 程序 员 ， 可 以 略 过 本 半 ， 因 为 它 介绍 了 
一 些 和 全 局 变量 、 数 据 类 型 以 及 改变 类 型 相关 的 、PHP 特 有 的 功能 。 


51 TE 


变量 (variable) 441 AU CI —MPAIN M en 
U RAAE, BE, Rr FTTR, ORR. BARAA PAR 
值 。 变 量 是 编程 的 基本 概念 ， 没 有 变量 ， 对 脚本 中 用 到 的 每 个 符 定 值 都 
需要 进行 直接 编码 。 如 下 的 直接 编码 语句 把 两 个 数字 加 在 一 起 并 且 显 示 
出 结 有 末 ， 这 解雇 了 一 个 简单 的 数学 问题 。 


echo (2 + 4}; 


ZAM, IR BOE IR XT AB EE SCRE AU 2 D4 A) All ce 2 2 LAA H o 
AY STE BR ll, BRAT AY DA a — BSR AS ES TR PE A, A 
如 3 和 5。 然 而 ， 这 种 编程 方法 很 元 座 ， 而 这 正 是 变量 的 用 武之 地 。 


变量 允许 我 们 为 运算 创建 模板 ， 例 如 ， 把 两 个 数 相 加 ， 而 不 用 关心 
变量 具体 所 表示 的 值 。 当 脚本 运行 的 时 候 ， 值 将 会 赋 给 变量 ， 可 能 通过 
用 户 输入 、 通 过 一 个 数据 库 查询 或 者 通过 脚本 中 之 前 的 另 一 个 操作 的 结 
果 获得 值 。 换 名 话说 ， 当 脚本 中 的 数据 很 可 能 改变 的 时 候 〈 不 管 是 在 脚 
本 的 生存 期 内 ， 还 是 当 它 传递 给 稍 后 使 用 的 另 一 个 脚本 时 ) ， 应 该 使 用 
变量 。 


一 个 变量 由 你 所 选取 的 一 个 名 字 以 及 前 面 的 一 个 美元 符号 〈$) 组 
成 。 变 量 名 可 以 包 侣 字母 、 数 字 以 及 下 划 线 C ， 但 不 能 包含 空格 。 
变量 名 必须 以 一 个 字母 或 一 个 下 划 线 开始 。 下 面 是 一 些 合法 的 变量 。 
$a; 
$a_longish_variable_name; 


$2453; 
SsleepyZZ27Z: 


变量 名 应 该 有 意义 而 且 要 风格 一 致 。 例 如 ， 如 果 你 的 脚本 要 处 理 名 字 (name) 和 密码 
(password) ， 不 要 为 名 字 创 建 一 个 名 为 $n 的 和 变量， 为 密码 创建 一 个 $p 变 量 ， 因 为 对 于 除了 你 
以 外 的 任何 人 ， 这 都 是 没有 意义 的 名 字 。 而 且 如 果 你 数 周 之 后 再 回顾 这 段 脚本 ， 你 可 能 会 把 
$n 当 作 是 “number” 而 不 是 “hame” 的 变量 ， 或 认为 $p 代 表 “page” 而 不 是 “password”。 如 果 一 个 合 
作者 想 要 修改 你 的 脚本 该 怎么 办 ? 他 怎么 知道 和 mn 和 $p 代 表 什 么 ?你 可 以 对 自己 的 脚本 中 的 变 
量 使 用 任何 命名 惯例 ， 只 要 名 字 是 具有 描述 性 的 并 且 遵 从 其 他 人 可 以 理解 的 某 种 模式 。 




















一 个 分 号 (5) (又 叫做 指令 终止 侍 ，instruction terminator) 用 来 结 
束 一 条 PHP 语 多。 上述 的 代 但 段 中 的 分 号 并 非 变 量 名 的 一 部 分 ， 而 是 用 
来 结束 声明 变量 的 语句 。 要 声明 一 个 变量 ， 你 只 需 将 其 包含 到 目 己 的 脚 
本 中 。 当 你 声明 一 个 变量 的 时 候 ， 通 常会 在 同一 条 语句 中 为 它 赋 一 个 初 
始 值 ， 如 下 所 示 。 


Gnumi = 8: 


$num2 23: 


上 述 代 码 行 声明 了 两 个 变量 ， 并 且 使 用 赋值 操作 符 〈=) 来 为 它们 
赋值 。 我 们 将 会 在 本 章 后 面 的 5.3 下 更 详细 地 了 解 赋值 。 在 为 变量 赋 信 
之 后 ， 我 们 可 以 将 变量 看 作 变 量 值 本 身 一 样 。 换 句 话说 。 


echo $num1; 


等 于 


echo 8; 


只 要 将 值 8 赋 给 了 了 $num1。 


5.1.1 全 局 变量 


变量 除了 命名 变量 的 规则 以 外 ， 还 有 一 些 有 关 可 用 性 的 规则 。 通 
前， 赋 给 一 个 变量 的 值 只 在 它 所 在 的 函数 或 脚本 中 有 效 。 例 如 ， 如 果 你 
有 一 个 scriptA.php 保 存 了 一 个 名 为 $name、 值 为 joe 的 变量 ， 如 果 想 要 创 
建 一 个 scriptB.php， 它 也 使 用 $name 变 量 ， 我 们 可 以 给 第 二 个 $name 爸 量 
赋值 jane， 而 蛙 不 影 响 scriptA.php 中 的 变量 。 对 于 每 个 脚本 ，$name 变 量 
的 值 都 是 局 部 的 〈local) ， 并 且 赋 给 的 值 相 互 之 间 是 独立 的 。 


然而 ， 我 们 也 可 以 在 一 个 脚本 或 者 函数 中 把 $name 变 量 定 义 为 全 局 
的 〈global) 。 如 果 在 scriptA.php 和 scriptB.php 中 ， 将 $name 变 量 定义 为 
一 个 全 局 变量 ， 并 且 ， 这 两 个 脚本 彼此 连接 〈 也 吏 是 说 ， 一 个 脚本 调用 
男 一 个 脚本 ， 或 者 包含 男 一 个 脚本 ) ， 那 么 对 于 共 至 的 $name 变 量 将 只 
有 一 个 值 。 全 局 变量 作用 工 的 例子 将 在 第 7 章 详细 说 明 。 


5.1.2 ”超人 全 局 变量 


除了 自己 创建 的 全 局 变量 ，PHP 还 有 几 个 叫做 超 全 局 变量 
(superglobal〉 的 预定 义 的 变量 。 这 些 变 量 忆 是 存在 的 ， 并 且 它 们 的 值 
也 总 是 对 所 有 的 脚本 可 用 的 。 如 下 的 每 个 超 全 局 变量 ， 实 际 上 都 是 其 他 
变量 的 一 个 数组 。 


。$_GET 包 含 了 通过 GET 方 法 提供 给 一 个 脚本 的 任何 变量 。 

。$_POST 包 含 了 通过 POST 方 法 提供 给 一 个 脚本 的 任何 变量 。 

。$_COOKIE 包 含 了 通过 cookie 提 供给 一 个 脚本 的 任何 变量 。 

。$_FILES 包 含 了 通过 文件 上 传 提 供给 一 个 脚本 的 任何 变量 。 
。$_SERVER 包 含 了 像 标 头 、 文 件 路 径 和 脚本 位 置 等 信息 。 

。$_ENV 包 含 了 作为 服务 器 环境 的 一 部 分 提交 给 一 个 脚本 的 任何 变 


E, 


FA o 


。$_REQUEST 包 含 了 通过 GET、POST 或 COOKIE 输 入 机 制 提供 给 一 
个 脚本 的 任何 变量 。 
。 $_SESSION 包 含 了 在 一 个 会 话 中 当前 注册 的 任何 变量 。 


本 书 中 的 例子 将 会 在 可 能 的 情况 下 使 用 超 全 局 变量 。 在 脚本 中 使 用 
超 全 局 变量 对 于 创建 安全 的 应 用 程序 很 重要 ， 因 为 超 全 局 变量 减少 了 用 
户 注入 式 攻 击 进 入 到 脚本 的 可 能 性 。 通 过 编码 ， 可 以 让 脚本 只 接受 你 想 
要 的 内 容 ， 并 且 按 照 你 定义 的 方式 〈 例 如 ， 从 使 用 POST 方 法 的 一 个 表 
单 或 从 一 个 会 话 ) 来 接受 ， 可 以 消除 一 些 由 于 松 敌 地 编写 脚本 而 引 友 的 


问题 。 


5.2 ”数据 类 型 


不 同 的 数据 类 型 占用 不 同 的 内 存量 ， 并 且 在 一 个 脚本 中 操作 它们 的 
时 候 可 能 区 列 对 每 它们 。 一 些 编程 语言 因此 要 求 程序 员 提 前 声明 一 个 变 
量 所 要 包含 的 数据 的 类 型 。 相 反 ，PHP 是 类 型 宽松 的 语言 ， 这 意味 着 它 


将 在 数据 被 赋 给 每 个 变量 的 时 候 才 确定 数据 基 型 。 


这 种 目 动 类 型 真是 神 福 相依 。 一 方面 ， 它 意味 痢 变 量 可 以 元 活 地 便 
用 ， 例 如 ， 一 个 变量 可 以 存储 字符 串 ， 并 且 随 后 它 可 以 在 脚本 中 存储 整 
数 或 东 些 其 他 数据 类 型 。 态 一 方面 ， 在 较 六 的 脚本 中 ， 如 末 你 特别 硕 理 
一 个 变量 存储 条 种 数据 类 型 而 它 所 存储 的 东西 完全 个 同 的 话 ， 这 种 灵活 
性 可 能 会 导致 问题 。 例 如 ， 假 设 你 在 编写 用 来 操作 一 个 数组 变量 的 代 
人 码 ， 如 来 所 讨论 的 变量 是 一 个 数字 值 ， 而 不 十 数组 值 ， 当 代码 试图 在 这 
个 变量 上 执行 特定 于 数组 的 操作 的 时 候 ， 束 会 及 生 错 误 。 


表 5-1 给 出 了 PHP 中 可 用 的 8 种 标准 的 数据 类 型 。 


表 5-1 标准 数据 类 型 


Float 或 Double 3.234 —/ME a 








String “hello” 字符 的 一 个 集合 


Resource 对 一 个 第 三 方 资源 〈 如 数据 库 ) 的 引用 


Resource 类 型 经 和 党 由 人 处理 外 部 应 用 程序 或 文件 的 函数 返回 。NULL 
类 型 是 为 了 那些 已经 声明 但 还 没有 赋值 的 变量 保留 的 。 





PHP 有 几 个 函数 可 以 测试 一 个 变量 的 特定 类 型 的 合法 性 ， 实 际 上 ， 
每 个 类 型 都 有 一 个 函数 。 这 是 一 组 is_* 消 数 ， 例 如 is_bool0 测 试 一 个 给 
定 的 值 是 否 是 布尔 值 。 程 序 清单 5.1 把 几 个 不 同 的 数据 类 型 分 配给 单个 
的 变量 ， 然 后 使 用 相应 的 is_* 函 数 来 测试 这 个 变量 。 代 码 中 的 注释 显示 
脚本 处 理 到 哪里 了 。 


提示 : 
天 于 调用 函数 ， 我 们 将 在 第 7 章 了 解 更 多 信息 。 


程序 清单 5.1 测试 一 个 变量 的 区 型 


1: <?php 

2: $testing; // declare without assigning 

3: echo "is null? ".is null($testing); // checks if null 

4: echo "<br/>"; 

5: $testing = 5; 

6: echo "is an integer? ".is int($testing); // checks if integer 
7: echo "<br/>"; 

8: $testing = "five"; 

9: echo "is a string? ".is_string($testing}; // checks if string 


10: echo "<br/>"; 

11: $testing = 5.024; 

12: echo "is a double? ".is double($testing); // checks if double 

ie eho “pris 

14: $testing = true; 

15: echo "is boolean? ".is bool($testing); // checks if boolean 

16: echo "<br/>"; 

17: $testing = array{'apple', 'orange', 'pear'); 

18: echo "is an array? ".is array($testing); // checks if array 

19: echo "<br/>"; 

20: echo "is numeric? ".is numeric($testing); // checks if is numeric 
ala echo *<prj/>"; 

22: echo "is a resource? ".is resource($testing); // checks if is a resource 
23: echo "<br/>"; 

24: echo "is an array? ".is_ array($testing); // checks if is an array 
25: echo "<br/>"; 

26: +> 


把 上 述 代 码 保存 到 一 个 名 为 testtype.php 的 文本 文件 中 ， 并 且 把 这 个 
文件 放 入 到 你 的 Web 服 务 右 文档 根 日 录 下 。 当 你 通过 Web 浏 览 右 访问 这 
个 脚本 的 时 候 ， 会 产生 如 下 的 输出 。 


is null? 1 

is an integer? 1 
1s a string? 1 
is a double? 1 
1s boolean? 1 

is an array? 1 
is numeric? 

is a resource? 
is an array? 1 


我 们 在 第 2 行 声 明了 $testing 变 量 ， 却 没有 为 它 赋值 ， 因 此 ， 当 我 们 
在 第 3 行 测试 这 个 变量 看 它 是 否 为 空 的 时 候 〈 使 用 is_nulO0) ， 结 果 是 


1(true). 


yy. 
YER: 


WARP OARCE PHARMA AR, SEVIS FT ON RI BCAA YIN AR, 
将 会 看 到 如 下 的 一 条 注意 事项 。 


Notice: Undefined variable: testing in /path/to/testtype.php on line 3 


当 你 使 用 php-development.ini 而 不 是 php-production.ini 的 时 候 ， 注 意 会 默认 地 打开 。 调 试 脚 
ASIEN (FT FRE RIE AHS 


(LAA J Stestinge ane Za, A= SAEZ J $testing, “A 
Ja» EHM Ais * RAZOR. TESST, ERUR T 
$testing tE, Re TRARA. fa, RITE WT eae 
WEKA NAAMA. ÆR8T, —DTITRIRA I Stesting tE, X 
征 一 个 字符 的 集合 。 当 你 在 自己 的 脚本 里 使 用 字符 串 的 时 候 ， 它 们 总 是 
用 单 引 号 或 者 双 引 号 插 起 来 〈' 或 ") 。 在 第 11 行 ， 一 个 双 精 度 浮 点 数 赋 
给 了 $testing 变 量 ， 这 是 一 个 序 点 数 〈 即 ， 一 个 包 舍 小 数 点 的 数字 ) 。 在 
第 14 行 ， 一 个 布尔 值 赋 给 了 9$testing 变 量 ， 它 的 值 是 两 个 指定 值 〈true 或 
false) 中 的 一 个 。 在 第 17 行 ， 使 用 array0O 函 数 创 建 了 一 个 数组 ， 我 们 将 
在 第 8 章 详细 学 习 数 组 。 这 个 特定 的 数组 包含 了 3 个 数据 项 ， 并 且 脚 本 正 
确 地 报告 $testing 具 有 “数组 ”类 型 。 


从 第 20 行 到 脚本 的 末尾 ， 没 有 值 乍 新 赋 给 $testing， 只 有 类 型 测试 。 
Fi 2047 Ml 282247 a} Al dll A Stesting Ae — TAF IRA, WRN 
BLASS IAAP aa. AAAS ESR 2477 YR ila Stesting ze Ae TAH, 
OR ce Wt MAN « 


5.2.1 ”使 用 settype() 来 改变 变量 的 数据 类 型 


PHP 也 提供 了 疯 数 settype()， 用 来 改变 一 个 变量 的 类 型 。 要 使 用 这 
个 函数 ， 你 需要 在 括 写 中 放 入 要 修改 类 型 的 变量 以 及 要 改变 的 目标 类 
AY, FFA A Sh Soc, OR PTR. 


settype($variabletochange, ‘new type'); 


Fer in ¥45.290183.14 (CNF MBO FAITE A E P rte BUY 
4 种 其 他 标准 类 型 。 


程序 清单 5.2 ”使 用 settype0) 修 改 一 个 变量 的 类 型 


1 <?php 

2: $undecided = 3.14; 

3: echo "is ".$undecided." a double? ".is double($undecided)."<br/>"; // double 

4: settype($undecided, 'string'); 

5: echo “is ",.$undecided." a string? ".1S string($undecided)."<br/>"; // string 

6: settype($undecided, 'integer'); 

7 echo "is ".$undecided." an integer? ".is integer($undecided)."<br/>"; /| 
integer 

8: settype($undecided, 'double'); 

9: echo "is ".$undecided." a double? ".is double($undecided)."<br/>"; // double 

10: settype($undecided, ‘bool'); 

11: echo "is ".$undecided." a boolean? ".is bool($undecided)."<br/>"; // boolean 

128 "E> 





根据 PHP 手 册 ， 在 浮 点 数 的 情况 下 返回 “double”， 而 不 是 直接 返回 “float*。 我 们 所 看 到 的 
情况 也 确实 如 此 。 


在 每 个 实例 中 ， 我 们 使 用 了 相应 的 is_* 孙 数 来 确 你 新 的 数据 类 型 ， 
并 且 还 使 用 echo 把 变量 $undecided 的 信和 输 出 到 浏 虎 规 。 当 我 们 在 第 6 行 把 
宇 符 吝 “3.14? 转 换 为 一 个 整数 时 ， 小 数 点 后 面 的 所 有 信息 都 永久 地 丢失 
了 。 这 殉 是 为 什么 当 我 们 在 第 8 行将 其 改 回 double 类 型 的 时 候 ， 


$undecided 中 包含 的 值 是 3。 最 后 ， 在 第 10 行 ， 我 们 把 $undecided 转 换 为 
一 个 布尔 什 ， 任 何 非 0 的 数字 转换 为 布尔 值 时 都 会 变 为 tue。 当 我 们 在 

PHP 中 显示 一 个 布尔 值 的 时 候 ，true 显 示 为 1 而 false 显 示 为 一 个 空 字符 

早 ， 因 此 ， 在 第 11 行 ，$undecided 显 示 为 1。 


把 上 述 代 码 放 入 到 一 个 名 为 settype.php 的 文本 文件 中 ， 并 且 把 这 个 
文件 放 到 你 的 Web 服务 问 的 文档 根 目录 下 ， 妆 我 们 通过 Web 浏 贤 右 访问 
这 个 脚本 的 时 候 ， 会 产生 如 下 的 输出 。 
is 3.14 a double? 1 
is 3.14 a string? 1 
1s 3 an integer? 1 


is 3 a double? 1 
1s 1a boolean? 1 


提示 : 


你 将 不 会 看 到 像 前 面 小 节 中 所 见 到 的 那样 一 条 关于 未 定义 变量 的 注意 ， 因 为 在 这 段 脚本 
的 开始 ， 定 义 了 $undecidedis 变 量 并 给 它 赋 了 一 个 值 。 


5.2.2 ”通过 类 型 转换 改变 变量 的 数据 类 型 


使 用 settype() 改 变 一 个 已 有 变量 的 类 型 和 使 用 类 型 转换 改变 变量 类 
型 的 主要 区 别 在 于 ， 类 型 转换 会 生成 一 个 拷贝 ， 而 保持 原来 的 变量 不 
动 。 要 通过 类 型 转换 来 改变 类 型 ， 我 们 首先 在 括号 中 ， 在 所 要 复制 的 变 
量 的 前 面 ， 给 出 一 种 数据 类 型 的 名 字 。 例 如 ， 如 下 的 一 行 代码 创建 了 
$originalvar 变 量 的 一 个 副本 ， 它 其 有 一 个 指定 的 类 型 〈 整 数 ) 和 一 个 新 
的 名 字 $newvar。$originalvar 变 量 仍然 可 用 ， 并 用 仍 保 留 原 来 的 类 型 。 
$newvar 是 一 个 全 新 的 变量 。 


Gnewvar = (integer} $originalvar 


程序 清单 5.3 展 示 了 通过 类 型 转换 改变 数据 类 型 的 例子 。 
程序 清单 5.3 ”对 一 个 变量 进行 类 型 转换 


<?php 

Sundecided = 3.14; 

$holder = (double) $undecided; 

echo "is ".$holder." a double? ".is double($holder)."<br/>"; // double 
$holder = (string) $undecided; 

echo "is ".$holder." a string? ".is string($holder)."<br/>"; // string 
$holder = {integer} $undecided; 

echo "is ".$holder." an integer? ".is integer($holder)."<br/>"; // integer 
$holder = (double) $undecided; 


19: echo “is ".$holder." a double? ".is double($holder)."<br/>"; // double 
11: $holder = {boolean} $undecided; 


12: echo "is ".$holder." a boolean? ".is bool($holder)."<br/>"; // boolean 
13: echo "“<hr/>"; 
14: echo "original variable type of $undecided: "; 
15: echo gettype($undecided); // double 
ie 


OmnN DOoahWNh — 


我 们 不 会 真 的 改变 $undecided 变 量 的 类 型 ， 它 在 这 个 脚本 中 始终 人 
持 为 double 类 型 》 Fs 15 行 说 明 了 E 》 在 那里 》 我 们 使 用 settype() pk] 
数 来 确定 $undecided 的 类 型 。 


提示 : 








不 要 使 用 gettype0 来 测试 条 个 变量 的 数据 类 型 ， 尽 管 在 这 里 是 这 么 做 的 ， 因 为 这 可 能 很 悍 
并 且 访 函数 可 能 在 未 来 的 版 本 中 被 公用。 在 正式 的 情况 下 ， 使 用 is_* 函 数 来 测试 变量 的 数据 类 
型 。 这 里 使 用 gettype() 函 数 是 为 了 便于 说 明 。 


实际 上 ， 通 过 对 $undecided 进 行 类 型 转换 ， 我 们 创建 了 一 个 副本 ， 
然后 将 其 转换 为 在 类 型 转换 时 所 指定 的 类 型 ， 开 且 和 存储 a 到 变量 $holder 
中 。 这 个 类 型 转换 肖 先 发 生 在 第 3 行 ， 随 后 还 友 生 在 第 5 行 、 第 7 行 、 第 9 
行 和 第 11 行 。 由 于 我 们 只 是 使 用 $undecided 的 一 个 副本 而 不 是 最 初 的 变 
量 ，$undecided 不 会 失去 其 最 初 的 值 ， 这 和 在 程序 清和 蛙 5.2 中 第 6 行 


$undecided 杰 量 的 类 型 从 字符 串 转 换 为 整数 的 情况 不 一 样 。 


把 程序 清单 5.3 的 内 容 放 入 到 一 个 名 为 casttype.php 的 文本 文件 ， 然 
后 把 这 个 文件 放 入 到 你 的 Web 服 务 右 文档 根 目 录 下 。 当 你 通过 Web 浏 贤 
占 访 问 这 个 脚本 的 时 候 ， 它 会 产生 如 下 的 输出 。 
is 3.14 a double? 1 
is 3.14 a string? 1 
1s 3 an integer? 1 
is 3.14 a double? 1 


is 1 a boolean? 1 
original variable type of 3.14: double 


现在 你 已 经 了 解 了 如 何 使 用 settypeO 〇 函数 或 通过 类 型 转换 把 一 个 变 
量 从 一 种 类 型 改变 为 为 一 种 类 型 ， 沽 虑 一 下 为 什么 这 种 转换 可 能 会 有 
用 。 这 并 不 是 一 个 必须 经 第 使 用 的 过 程 ， 因 为 在 脚本 内 容 需 要 类 型 转变 
的 情况 下，PHP 会 目 动 为 你 进行 变量 类 型 转换 。 然 而 ， 目 动 的 类 型 转换 
是 临时 的 ， 并 且 你 可 能 想 让 一 个 变量 持久 地 保存 一 种 特定 的 数据 类 型 ， 
因此 ， 束 需要 明确 地 改变 类 型 。 


例如 ， 一 个 用 户 从 一 个 HTML 表 单 中 输入 数字 ， 这 个 数字 将 要 贷 脚 
本 作为 “字符 种 ?并 型 使 用 。 如 于 试图 把 两 个 字符 串 相 加 ， 由 于 它们 包含 
数字 ， 在 相 加 的 时 候 ，PHP 会 帮忙 把 这 些 字 符 串 转换 为 数字 。 因 此 


38cm + "40cm" 


将 会 产生 一 个 结 来 70。 





通用 术语 “数字 ”在 这 里 指 的 是 整数 和 浮 点 数 。 如 果 用 户 输入 是 浮 点 数 的 形式 ， 并 且 相 加 
的 字符 串 为 "3.14cm” 和 “4.12cm”， 答 案 将 会 是 7.26。 在 把 字符 串 类 型 转换 为 整数 或 浮 点 数 的 时 
候 ，PHP 将 会 忽略 任何 非 数 字 字 符 ， 字 符 串 将 会 被 截断 ， 并 且 ， 从 第 一 个 非 数 字 字 符 开 始 同 后 








HATA PT aR A Ak, FR 30cm RN”, FAT AR “GFt2in” Ae RK, ALAA Rl 
余部 分 都 将 计算 为 0。 





你 可 能 想 要 目 己 清空 用 户 输 入 ， 并 且 在 脚本 中 用 特殊 的 方式 使 用 
它 。 假 设 己 经 要 求 用 户 提交 一 个 数字 ， 我 们 可 以 通过 声明 一 个 变量 并 且 
把 用 户 的 输入 赋 给 它 来 模拟 这 种 情况 。 
$test = "30cm"; 

正如 你 所 见 到 的 ， 用 户 已 经 为 他 们 的 数字 添加 了 单位 ， 用 户 输 入 的 
是 “30cm”， 而 个 是 “30”。 可 以 通过 将 其 类 型 转换 为 一 个 整数 来 确保 用 三 
得 入 是 整齐 的 。 


$newtest = (integer) $test; 
echo "Your imaginary box has a width of $newtest centimeters."; 


最 终 的 输出 如 下 。 
Your imaginary box has a width of 3@ centimeters. 

如 果 用 尸 的 输入 没有 进行 类 型 转换 ， 并 且 当 显示 有 关 一 个 盒子 的 宽 
上 度 的 语句 时 ， 最 初 的 变量 $test 的 值 将 蔡 代 $newtest， 结 果 如 下 。 
Your imaginary box has a width of 30cm centimeters. 

这 个 输出 显得 奇怪 ， 实 际 上 ， 它 看 上 去 只 是 重复 了 未 经 整理 过 的 
(原始 的 ) 用 户 输 入 。 


5.2.3 TMAR 


为 什么 知道 变量 的 类 型 可 能 会 有 用 呢 ? 在 编程 中 经 常会 有 这 样 的 情 
况 ， 传 递 给 你 的 数据 是 来 自 于 故 外 一 个 来 源 。 在 第 7 草 中 ， 你 将 会 学 习 
如 何在 目 己 的 脚本 中 创建 函数 ， 以 及 数据 弟 第 在 一 个 或 多 个 函数 之 间 传 


递 ， 因 为 它们 可 以 以 参数 的 形式 从 调用 代码 接 党 信息 。 对 于 要 使 用 给 定 
数据 类 型 的 函数 ， 首 和 完 验 证 函数 被 给 定 的 值 具有 正确 的 数据 类 型 是 一 个 
人 不错 的 主意 。 例 如 ， 期 每 拥有 一 个 “资源 ”类 型 的 数据 的 一 个 函数 ， 妆 传 
RES TE PE EAT ERR, ETC IE i LF AY 


5.3 ”操作 人 符 和 表达 去 


根据 我 们 目前 学 习 的 知识 ， 我 们 可 以 把 数据 赋 给 变量 ， 甚 至 可 以 了 
解 变 量 的 数据 类 型 并 改变 它 。 然 而 ， 际 非 你 可 以 操作 已 经 存储 的 数据 ， 
侍 则 ， 一 种 编程 语言 还 并 不 是 非 常 有 用 。 操 作 和 人 符 (operator) 就 是 用 来 
操作 存储 在 变量 中 的 数据 的 符号 ， 从 而 使 得 以 下 情况 成 为 可 能 : 用 一 个 
值 或 多 个 值 来 产生 一 个 新 的 值 ， 或 者 在 一 个 条 件 下 检查 数据 的 有 效 性 以 
便 确 定 下 一 个 步 又 等 等 。 操 作 从 在 其 上 操作 的 值 叫做 操作 数 
(operand) 。 


提示 : 





操作 符 是 一 个 从 写 或 者 一 系列 的 答 写 ， 当 它 用 来 连接 值 的 时 候 ， 执 行 一 个 操作 并 且 通 当 
会 产生 一 个 新 值 。 操 作 数 是 用 来 和 一 个 操作 符 连 接 的 一 个 值 。 一 个 操作 符 通 种 有 两 个 或 更 多 
个 操作 数 。 





在 下 面 的 简单 例子 中 ， 两 个 操作 数 和 一 个 操作 从 组 合 到 一 起 产生 一 
个 新 的 值 。 
(4 + 5) 

整 效 4 和 5 十 操 作 数 。 这 些 操作 数 通 过 一 个 加 喜 操 作 符 〈+) 来 操 
作 ， 产 生 整 数 9。 操 作 符 几乎 总 是 位 于 两 个 操作 数 之 间 ， 尽 管 你 会 在 本 
草 的 后 面 看 到 人 少数 例外 情况 。 


操作 数 和 一 个 操作 符 组 合 到 一 起 得 到 一 个 结 采 就 叫做 表达 式 
(expression) 。 尺 官 操作 从 及 其 操作 数 构 成 了 表达 式 的 基础 ， 但 一 个 


表达 式 不 一 定 要 包含 操作 符 。 实 际 上 ，PHP 中 的 表达 式 定 义 为 可 以 用 作 
一 个 值 的 任何 事物 。 这 包括 像 654 这 样 的 整数 常数 ， 像 $user 这 样 的 变 
量 ， 以 及 像 gettype(O 这 样 的 函数 调用 。 例 如 ， 表 达 式 (445) 由 两 个 表达 
A“ (4 和 “5) ”以 及 一 个 操作 符 “+” 组 成 。 当 一 个 表达 式 产 生 一 个 值 
时 ， 通 第 说 求 得 访 值 。 也 束 是 说 ， 当 考 上 处 到 所 有 的 子 表达 式 的 时 候 ， 表 
达 式 可 以 看 做 好 像 是 该 值 本 身 的 代码 。 在 这 个 例子 中 ， 表 达 式 (4+5) 
求 得 值 9。 


提示 : 


一 个 表达 式 可 以 是 函数 、 值 和 求 值 的 操作 人 符 的 任 音 组合。 首要 的 原则 是 ， 如 末 你 可 以 将 
其 作为 一 个 值 使 用 ， 它 就 是 一 个 表达 式 。 


既然 我 们 已 经 解决 了 原理 性 问题 ， 现 在 可 以 看 看 操作 符 在 PHP 程 序 
设计 中 的 一 般 用 法 。 
5.3.1 IERIE IT 

在 以 前 的 例子 中 声明 一 个 变量 的 时 候 ， 我 们 已 经 见 到 过 赋值 操作 符 
的 几 次 使 用 ， 赋 信 操 作 符 由 单个 字符 组 成 ， 即 =。 赋 什 操 作 人 符 取 右 操作 
数 的 值 并 将 其 赋 给 左 操作 数 ， 示 例如 下 。 
$name = "Jimbo"; 

$name 变 量 现在 包含 了 字符 串 “*jimbo”。 这 个 结构 也 是 一 个 表达 式 ， 
只 不 过 第 一 眼看 上 去 好 像 同 全 操作 符 上 只 是 改变 了 $name 变 量 的 仁 ， 而 没 
有 产生 一 个 新 的 伍 。 实 际 上 ， 使 用 赋值 操 作 符 的 语句 总 是 求 得 右 操作 数 
的 值 的 一 个 副本 。 因 此 ， 下 述 代码 


echo $name = "Jimbo"; 


=o 


[eX at ase AN AF AB “jimbo”, [AJ IAFL “jimbo” ul Za $name = 
5.3.2 ”算术 操作 和 从 


算术 操作 符 做 我 们 所 期 待 的 事情 ， 即 执行 算术 运算 。 表 5-2 列 出 了 
这 些 操 作 符 ， 并 举例 说 明 它 们 的 用 法 和 结 


表 5-2 算术 操作 符 
操作 和 从 名 称 AN Bill 25 AR 


-减法 


/除法 10/3 3.3333333333333 





加 法 操作 从 把 右 操 作 数 增加 a 到 左 操 作 数 上 ; 减法 操作 从 把 右 操 作 数 
从 左 操 作 数 上 减 去 ， 除法 操作 符 用 左 操 作 数 除 以 右 操 作 数 ， 乘 法 操作 符 
用 芯 操 作 效 滋 以 右 操作 数 ， 模 除 操 作 符 返 回 和 元 操作 数 除 以 右 操 作 效 的 余 
数 。 


5.3.3 ”连接 操作 符 


连接 操作 符 用 一 个 句点 〈.) 表示 。 它 把 两 个 操作 数 都 当 作 是 字符 
串 ， 把 右 操 作 数 附加 到 左 操 作 数 上 。 如 
"hello"." worla" 

1K |e] 
"hello world" 

注意 ， 单 词 之 间 最 终 有 一 个 空格 ， 这 是 因为 在 第 二 个 操作 数 (是 “ 
world” 而 个 是 “world”) 的 前 面 有 一 个 空格 。 连 接 操 作 符 逐 字 把 两 个 字符 
串 连 接 起 来 而 不 深 加 任何 空 日 。 因 此 ， 如 果 你 想 要 连接 两 个 头 部 和 尾部 
部 没有 衬 格 的 字符 串 的 话 ， 例 如 
"hello". "world" 

会 得 到 如 下 的 结果 。 
"helloworld" 

不 管 和 连接 操作 符 一 起 使 用 的 操作 数 是 什么 数据 类 型 ， 它 们 都 会 被 
当 作 和 字符 串 对 每 ， 并 且 结 果 也 忆 是 字符 串 类 型 。 在 本 书 中 我 们 会 频 莹 地 
用 到 连接 操作 符 ， 妆 我 们 十 要 把 条 种 类 型 的 表达 式 结 末 和 一 个 字符 串 组 
合 到 一 起 的 时 候 使 用 ， 示 例如 下 。 


$cm = 212; 
echo "the width is ".($cm/1@0)." meters"; 


5.3.4 ”复合 赋值 操作 从 


里 然 只 有 一 个 真正 的 赋值 操作 从 ， 但 是 PHP 拓 供 了 一 些 复合 赋值 操 
作 符 来 改变 左 操 作 数 并 返回 一 个 结果 ， 同 时 也 修改 了 变量 最 初 的 值 。 按 
照 规 则 ， 操 作 符 使 用 操作 数 但 不 会 改变 其 最 初 的 什 ， 但 是 复合 赋值 操作 


FETT MM SIX PI. SEMEL PR VE TE “Pa EPR YE 9 J TER A “1 
号 构成 。 复 合 赋值 操作 符 为 你 省 云 了 在 脚本 中 分 两 步 来 使 用 两 个 操作 符 
的 麻烦 。 例 如 ， 如 朱 有 一 个 值 为 4 的 变量 ， 并 且 息 要 再 把 这 个 人 加 4， 你 
可 能 会 看 到 如 下 实现 。 


$x 
$x 


4; 
$x + 4; // $x now equals 8 


然而 ， 我 们 也 可 以 使 用 一 个 复合 赋值 操作 符 (+=) 来 加 和 并 返回 新 
> U FER. 


$x = 4; 
$x += 4; // $x now equals 8 


每 个 复 术 操作 符 包 丘 连 接 操作 符 都 有 相应 的 复合 赋值 操作 符 。 雪 5- 
3 列 出 了 这 些 复合 赋值 操作 符 并 给 出 了 其 用 法 示例 。 


表 5-3 一 些 复 合 赋值 操作 符 


Be 作 FF 


$x=$x%5 





= $x.=“test” $x=$x.“test” 





表 5-3 中 的 每 个 例子 使 用 石 操 作 数 的 值 来 改变 $x 的 值 。 此 后 再 用 到 
SX HY EY (Bore S| Fr. AN BIEN R 


$x = 4; 

$x += 4; // $x now equals 8 
$x += 4; // $x now equals 12 
$x -= 3; // $x now equals 9 


这 些 操 作 符 将 会 在 本 书 的 很 多 脚本 中 用 到 。 妆 你 想 创建 一 个 动态 文 
AS ERE, KRWAWE BIS SEL BR VETS © UR “ASSP A 
符 串 添加 内 容 ， 人 例如， 动态 地 构建 HIML 代 码 来 显示 一 个 表 ， 是 复合 赋 
值 操 作 和 从 用 法 的 主要 例子 。 


5.3.5 ”自动 增加 和 减少 一 个 整 型 变量 

用 PHP 编 码 的 时 候 ， 你 将 经 常会 发 现 有 必要 对 一 个 整 型 变量 加 1 或 
减 1。 当 你 计算 一 个 循环 的 识 数 的 时 候 ， 通 稼 需要 这 人 么 做 。 你 已 经 学 习 
了 这 么 做 的 两 种 方式 ， 即 使 用 加 法 操作 符 来 增加 $x 的 值 : 


$x = $x + 1; // $x is incremented by 1 


或 者 使 用 一 个 复合 赋值 操作 符 。 


$x += 1; // $x is incremented by 1 


在 这 两 个 例子 中 ， 狐 的 值 都 赋 给 了 $x。 因 为 这 种 日 动 增加 或 减少 的 
表达 式 很 常见 ，PHP 提 供 了 某 些 专门 的 操作 和 从 以 允许 你 对 一 个 整 型 变量 
增加 或 减少 整数 第 量 1, FER ee A. Reverie Ae 
(post-increment) 和 后 目 减 Cpost-decrement) 。 后 目 增 操作 符 包 含 跟 在 
变量 名 后 面 的 两 个 加 号 ， 如 下 上 所 示 。 


$x++: // $x is incremented by 1 

这 个 表达 式 把 变量 $x 所 表示 的 值 增 加 了 1。 以 同样 的 方式 使 用 两 个 
减 号 从 与 将 会 减 小 该 变量 的 值 。 
Gx--; // $x is decremented by 1 

如 末 和 一 个 条 件 操作 符 一 起 使 用 后 目 增 或 后 目 减 操作 符 ， 那 么 ， 只 
有 在 第 一 个 操作 完成 以 后 操作 数 才 会 税 修 改 。 


$x 
$y 


3; 
$xt+ + 3; 


在 这 种 情况 下 ，$y 首 先 变 为 6(3 + 3)， 然 后 $x 增加 1。 


在 某 些 情况 下 ， 在 一 个 测试 表达 式 中 ， 你 可 能 希望 在 测试 执行 之 前 
把 一 个 变量 增加 或 减少 1。PHP 为 此 提供 了 前 目 增 (pre-increment〉 和 前 
目 减 Cpre-decrement) 操作 符 。 这 两 个 操作 和 从 按照 与 后 目 增 和 后 自 减 操 
作 和 从 相同 的 方式 工作 ， 但 是 ， 它 们 把 加 号 或 减 写 放 在 了 变量 的 前 和 耐 。 


++$x; // $x is incremented by 1 
- -$x; // $x is decremented by 1 


如 果 这 些 操 作 符 用 作 一 个 测试 表达 式 的 一 部 分 ， 变 量 加 1 将 会 在 测 
试 执行 之 前 执行 。 例 如 ， 在 下 面 的 代码 段 中 ， 在 测试 4 是 售 小 于 4 之 
前 ， 它 先 增加 1。 


$x = 3 
++$x < 4: // false 


测试 表达 式 返 回 false， 因 为 $x 首先 加 1 变 为 4， 而 4 不 再 小 于 4 了 了。 


5.3.6 ”比较 操作 符 


比较 操作 符 对 使 用 它们 的 操作 数 进行 比较 测试 ， 并 且 如 果 测 试 成 功 

返回 布尔 值 ttue， 如 果 测 试 失 败 返 回 布尔 值 false。 当 在 脚本 中 使 用 控制 

结构 的 时 低 ， 例 如 站 和 while 这 样 的 语句 ， 这 种 类 型 的 表达 式 很 有 用 。 我 
们 将 会 在 第 6 章 介绍 过 和 while 语 句 。 


例如 ， 要 测试 $x 中 包含 的 值 是 否 小 于 5， 我 们 可 以 使 用 小 于 操作 符 
作为 表达 式 的 一 部 分 ， 示 例如 下 。 


$x < 5 


如 果 $x 包 含 的 值 是 3， 这 个 表达 式 的 值 将 为 trtue。 如 果 $x 包 仿 的 值 是 
7， 这 个 表达 式 将 会 得 到 false。 


表 5-4 列 出 了 比较 操作 从。 


表 5-4 比较 操作 符 


Jem one | a 





> AT 左边 大 于 右边 $x>4 false 


小 于 或 等 于 | 左边 小 于 或 等 于 右边 $x<=4 


X ERRENTE E AEA OO PSE A I E SN Se PRET EH 
KERFI., ERER ETMEN ETS ZAI] 0 == 操作 从 
MAISE, MEEI. 53e” aafo === WA RASA 


相等 性 。 
5.3.7 1 H E FPR YE ITO E R 8 MARIA 


逻辑 操作 符 测 试 布尔 值 的 组 合 。 例 如 ，or 操 作 符 用 两 条 竖 线 (I) 或 者 
一 个 单词 or 来 表示 ， 如 果 左 操作 数 或 右 操 作 数 中 有 一 个 为 tue， 结 末 返 
器 布尔 值 true。 


true || false 





这 个 表达 式 返 回 true。 


and ENH ADERIT R 表示 或 者 直接 使 用 单词 and 来 表 
示 ， 如 果 两 个 操作 数 都 是 true， 它 天 返回 布尔 值 true。 


true && false 


XP FIA TUB II AHR (false. MART REREH E FRE FORM 
2 


ADR Hs RAIN PS Be PIA TOR FB POR ES 


例如 下 。 


(Sx > 2) && ($x < 15) 


如 来 $x 包含 了 一 个 大 于 2 而 小 于 15 的 值 ， 返 回 布尔 值 trtue。 妆 比较 表 
达 式 的 时 候 可 以 使 用 括 写 ， 以 便 代 人 码 更 容易 阅读 并 能 够 表示 表达 式 求 值 
的 优先 级 。 表 5-5 列 出 了 逻辑 操作 从。 


表 5-5 BEREN 


边 或 右边 为 true true | | false 
— 











i EE 
roe 
-ep 边 或 右边 为 tue， 但 不 都 为 true 
nae 


UR A BE sz BY PEAT A or#lland#e ye FF ABA PAS ARRAS, RECS ANY 
问题 。 答 案 是 取决 于 操作 的 优先 级 ， jjbit E 


5.3.8 ”操作 符 优 先 级 





ai 


SVE PIR TH EA BRETT SN eR,» PHP 9] S20) fs A Zc 
PATER TN OS PAE “SPR TE TT HY SER EIA TL, WRIA EAT 
HY, PHP S| S22 a5] Albee. PON, ECs RE RPS ta EB 


4+ 5 


这 里 不 会 引起 混淆 ，PHP 只 是 把 4 和 5 相 加 。 但 是 ， 对 于 下 面 的 具有 
两 个 操作 符 的 代码 。 
4+5*2 


这 束 产 生 了 一 个 问题 。PHP 是 应 该 先 对 4 和 5 求 和 ， 然 后 把 结果 乘 以 
2， 得 到 18 呢 ?还 是 应 该 把 4 和 5 乘 以 2 的 结果 相 加 ， 得 到 14 呢 ?如 果 我 们 
只 是 简单 地 从 左 到 右 地 读 取 ， 前 一 个 结果 是 正确 的 。 然 而 ，PHP 对 不 同 
的 操作 符 附 加 了 不 同 的 优先 级 ， 并 且 由 于 乘法 操作 符 比 加 法 操作 符 具 有 
更 高 的 优先 级 ， 这 个 问题 的 第 二 个 解答 是 正确 的 ， 即 把 4 和 5 乘 以 2 的 结 
果 相 加 。 


然而 ， 你 可 以 在 表达 式 的 外 围 放置 一 对 括号 ， 从 而 覆盖 掉 操 作 符 本 
身 的 优先 级 。 在 下 面 的 代码 段 中 ， 加 法 表达 式 会 在 乘法 表达 式 之 前 计 


Ft o 


(4+ 5) * 2 


MNS BEAR ARIA TAH BRE FT I Ce PE, BEAT SORE 
代码 更 为 清楚 并 且 避 免 挥 一 些 错 误 〈 例 如 在 购物 车 应 用 中 对 错误 的 小 计 
计算 销售 税 ) ， 这 是 个 不 错 的 想法 。 下 面 是 本 章 所 介绍 的 操作 符 按 照 优 
乞 级 高 低 的 一 个 列表 “有 共有 较 高 优先 级 的 操作 符 列 在 前 面 ) 。 


正如 你 所 见 到 的 ，or 比 || 的 优先 级 低 ， 并 且 and 比 && 的 优先 级 低 ， 
因此 ， 可 以 使 用 较 低 优先 级 的 逻辑 操作 符 来 改变 一 个 复杂 测试 表达 式 的 
读 取 方式 。 在 下 面 的 代码 段 中 ， 两 个 表达 式 是 等 同 的 ， 但 是 后 者 更 容易 
理解 。 


$x and $y || $z 
$x && ($y || $z) 
更 进一步 地 考虑 ， 如 下 的 代码 段 更 容易 理解 。 
$x and ($y or $z) 
然而 ， 这 三 个 例子 都 是 等 同 的 。 
优先 级 的 顺序 是 PHP 中 && 和 and 都 存在 的 唯一 的 原因 。 对 于 || 和 
or， 也 是 如 此 。 在 大 多 数 情 况 下 ， 和 那些 依靠 这 些 操作 符 的 优先 级 顺序 


的 代码 比较 ， 使 用 括号 会 让 代码 更 清 杷 并 且 更 少 出 错 。 在 本 书 中 ， 我 们 
将 倾 回 于 使 用 更 为 归 见 的 | | ABBR VETS, FP ERSTE SOR BC ERE HE HY 


操作 符 有 顺序 。 


5.4 常量 


变量 为 存储 数据 提供 了 一 种 灵活 的 方式 ， 因 为 我 们 可 以 在 执行 脚本 
的 过 程 中 随时 改变 它们 所 存储 的 值 及 其 类 型 。 然 而 ， 如 果 你 想 要 使 用 一 
个 在 整个 脚本 执行 过 程 中 必须 保持 不 变 的 人 的话 ， 可 以 定义 并 使 用 一 个 
fi (constant) 。 必 须 使 用 PHP 内 建 的 defineO 国 数 来 创建 一 个 利 量 ， 
随后 这 个 常量 是 不 能 改变 的 ， 除 非 再 次 明确 地 define0 它 。 要 使 用 
define() 冰 数 ， 把 常量 的 名 字 以 及 你 希望 赋 给 它 的 值 放 入 到 括 与 中 ， 中 间 
FAS BT, ANB F o 


define("YOUR CONSTANT NAME", 42}; 


你 想 要 设置 的 这 个 值 可 以 是 一 个 数字 、 一 个 字符 串 或 者 一 个 布尔 
值 。 按 照 惯例 ， 营 量 的 名 字 应 该 大 写 的 ， 和 常量 只 能 使 用 常量 名 访问 ， 不 
需要 美元 特写。 程序 清 单 5.4 展 示 了 了 如何 定义 和 访问 一 个 常量 。 


程序 清单 5.4 定义 和 访问 一 个 各 量 


: <?php 

: deflne( THE YEAR", 2012 ) 

< ecno “Et 1s the year “a THE YEAR; 
?> 


Ooh 一 


提示 : 


使 用 营 量 的 时 候 ， 别 志 了 它们 可 以 用 在 脚本 中 的 任何 地 方 ， 例 如 包含 在 脚本 内 的 一 个 外 
Hl PAI BCH o 


注意 ， 我 们 在 第 3 行使 用 了 连接 操作 符 把 常量 所 存储 的 值 附加 到 字 


ITE “It is the year” 的 后 面 ， 因 为 PHP 个 会 区 分 一 个 剃 量 和 引 写 之 间 的 一 
个 字符 串 。 


把 上 述 代码 放 入 到 一 个 名 为 constant.php 的 文本 文件 中 ， 然 后 把 该 文 
件 放置 到 Web 服 务 右 的 文档 根 目录 下 。 当 你 明 过 Web 浏 览 右 访问 这 个 肢 
本 的 时 候 ， 会 得 到 如 下 的 输出 。 


It is the year 2012 


define() ek 2 tH AY LA pes 38 =P RB BOR AE FY E A 2 FP TAK OP 
大 小 与 。 默 认 情 况 下 ， 香 量 名 是 区 分 大 小 写 的 。 然 而 ， 通 过 把 true 参 数 
传递 给 defineO 函 数 ， 我 们 可 以 改变 这 一 行为 ， 因 此 ， 如 末 我 们 可 以 建立 
新 的 THE_YEAR 和 常量 ， 示 例如 下 。 


define( "THE YEAR", "2012", true}; 


我 们 可 以 访问 它 的 值 而 个 用 担心 大 小 号， 示例 如 下 。 


echo 上 he year; 
echo ThE_YeAr; 
echo THE YEAR; 


FTA WB TATE SSAA, FEENS, BY 
2012。 对 于 那些 使 用 我 们 的 代码 的 其 他 程序 员 ， 这 个 功能 可 以 使 得 脚本 
更 为 友好 一 些 ， 因 为 他 们 访问 我 们 已 经 定义 的 一 个 常量 的 时 候 ， 不 需要 
考虑 大 小 号。 但 是 为 一 方 甸 ， 退 党 其 他 的 常量 是 区 分 大 小 写 的 ， 这 可 能 
增加 了 而 不 是 减少 了 程序 员 的 混 消 ， 因 为 他 们 可 能 未 记 应 该 以 何 种 方式 
来 对 行 哪个 曲 量 。 除 非 你 有 足够 充分 的 理由 这 么 做 ， 否 则 的 话 ， 还 是 你 
持 钟 量 区 分 大 小 与 并且 便 用 大 写字 母 来 定义 它们 才 是 最 安全 的 ， 这 是 一 
个 便于 记 住 的 约定 ， 更 别 说 这 是 标准 方式 了 。 


FUE te PHP AIA eA HEA i se A, AE 
_ FILE_ R EIPHP5 2a PTR CENA. wE LINER [Al 
AMF SHUT EL. EATS ASH ay a ee A PS, A TTI Ae 
apatite MAN, FFA SRE EIA Se. BERTIE N 
量 的 完整 列表 ， 参 见 


http:/www.php.net/manual/en/language.constants.predefined.php 。 





你 也 可 以 通过 PHP_VERSION 第 量 来 了 解 PHP 的 哪个 成 本 在 解释 脚 
本 。 上 友 布 一 个 错误 报告 时 ， 如 采 需 要 把 版 本 信息 包含 在 脚本 输出 中 ， 这 
个 寓 量 会 很 有 用 。PHP_VERSION 第 量 是 一 个 预定 义 和 音量 〈 并 且 是 一 个 
WAF) 。 要 获取 保留 字 币 量 的 完整 列表 ， 人 参见 


http:/www.php.net/manual/en/reserved.constants.php 。 


5.5 ”小 结 


在 本 章 中 ， 我 们 介绍 了 了 PHP 语言 的 一 些 基 本 功能 。 了 解 了 变量 以 太 
如 何 使 用 赋值 操作 符 来 为 它们 赋值 ， 介 绍 了 变量 和 内 建 的 超 全 局 变量 的 
作用 域 。 本 半 还 介绍 了 操作 和 人 符 ， 并 且 介 绍 了 如 何 把 一 些 最 第 见 的 操作 和 从 
组 合 到 表达 式 中 。 最 后 ， 我 们 学 习 了 如 何 定 义 和 态 问 第 量 。 


既然 已 经 掌握 了 PHP 的 一 些 基 础 ， 下 一 章 将 带 你 真正 体验 PHP 编 
程 。 你 将 学 习 如 何 编写 脚本 ， 以 及 在 变量 、 表 达 式 和 操作 人 符 的 帮助 下 ， 
做 出 流程 判断 和 重复 执行 任务 。 


5.6 Q&A 
Q: 为 什么 知道 一 个 变量 的 数据 类 型 是 有 用 的 ? 


A: 一 个 变量 的 数据 类 型 往往 限制 了 你 可 以 对 它 做 什么 。 例 如 ， 你 
不 能 对 一 个 简单 的 字符 串 执 行 和 数组 相关 的 功能 。 类 似 的 ， 在 算术 计算 
中 使 用 一 个 变量 之 前 ， 可 能 想 要 确 你 它 包含 一 个 整数 或 者 浮 点 数 的 值 ， 
即便 在 此 情况 下 PHP 第 剃 改 变数 据 类 型 以 帮助 你 。 


Q: 命名 变量 的 时 候 应 该 遭 守 什么 惯例 ? 


A: 目标 应 该 总 是 确保 代码 多 于 阅读 和 理解 。 像 gab123245 这 样 的 一 
个 变量 ， 对 于 你 了 解 它 在 脚本 中 的 作用 没有 任何 帮助 ， 而 且 容 易 导 致 输 
入 错误 。 保 持 变 量 名 人 简单 而 且 具 有 摘 述 性 。 当 你 大 概 一 个 月 后 再 回头 来 
看 代码 的 时 候 ， 变 量 名 $f 可 能 对 你 没 多 大 意义 。 而 变量 名 $filename 应 访 
更 有 意义 。 


Q: 我 应 该 了 解 操作 和 从 优先 级 表 吗 ? 
A: 没什么 理由 说 不 应 该 去 了 解 ， 但 是 如 朱 有 更 重要 的 任务 ， 我 情 


愿 偷懒 过 去 。 在 定义 目 己 的 优先 级 顺序 的 时 候 ， 通 过 在 表达 式 中 使 用 括 
号， 你 可 以 使 自己 的 代码 更 容易 阅读 。 


5.7 KRAJ 


这 个 实践 练习 设计 用 来 帮助 你 预测 可 能 过 到 的 问题 、 复 习 已 经 学 过 
的 知识 ， 并 且 开 始 把 知识 用 于 实践 。 


a) Iii 日 
pra) AF yell 
1. 下 列 变 量 中 的 哪 一 个 不 是 合法 的 ? 
$a value submitted by a user 
$666666xyz 
$xyz666666 
$ counter 


$the first 
$file-name 


2. 下 面 这 段 代 码 的 输出 是 什么 ? 
$num = 33; 


{boolean} $num; 
echo $num; 


3. 如 下 语句 的 输出 是 什么 ? 


echo gettype('4"}; 


4. 如 下 代 人 码 段 的 输出 是 什么 ? 
$test val = 5.5466; 


settype($test_val, "integer'"); 
echo $test val; 


5， 如 下 的 哪 条 语句 没有 包含 一 个 表达 式 ? 
4; 


is int(44}; 
5/12; 


6， 问 题 5 中 的 哪 条 语句 包含 了 一 个 操作 符 ? 


7. 下 面 的 表达 式 将 返回 什么 值 ? 


5s 2 


返回 值 的 数据 类 型 是 什么 ? 


oy 


fi E 


1. 变量 名 $666666xVyz 不 合法 ， 因 为 它 疫 有 以 字母 或 下 划 线 开头 。 
变量 名 $the first 不 合法 ， 因 为 它 包 含 一 个 空格 。$file-name 也 不 合法 ， 
为 它 包 人 一 个 非 字 母 的 字符 C) 。 


2. 这 上 段 代 人 码 将 会 显示 整数 33。 类 型 转换 会 产生 存储 在 $num 中 的 值 
的 一 个 转换 后 的 副本 。 它 不 会 修改 实际 存储 在 变量 里 的 初始 值 。 


3. 这 条 语句 将 会 输出 字符 串 “string”。 


4. 这 上段 代 人 码 将 输出 值 5。 当 一 个 浮 操 数 转换 为 一 个 整数 的 时 候 ， 小 
DUS WE a abe EK 


5. 它们 都 是 表达 式 ， 因 为 它们 都 会 求 得 值 。 
6. 语句 5/12 包 售 一 个 除法 运算 符 。 


7. 这 个 表达 式 将 会 得 到 false， 这 是 一 个 布尔 值 。 


EA wel 


1. 创建 一 个 脚本 ， 人 至 少 包含 5 个 不 同 的 变量 。 用 不 同 数据 类 型 的 值 
填充 它们 ， 并 且 使 用 gettypeO 函 数 在 浏 贤 占 中 显示 出 每 种 类 型 。 


2. 给 两 个 变量 赋值 。 使 用 比较 操作 从 来 测试 第 一 个 值 是 舍 和 第 二 
个 值 相同 。 


。 与 第 二 个 值 相 同 。 

。 小 于 第 二 个 值 。 

大 于 第 二 个 值 。 

小 于 或 等于 第 二 个 值 。 


在 浏览 如 中 显示 出 每 次 测试 的 结果 。 


修改 赂 给 调试 变量 的 值 ， 并 且 再 炊 运 行 脚本 。 


6 PHP 的 流程 控制 功能 


在 本 重 中 你 将 学 习 : 


如 果 一 个 测试 表达 式 结 果 为 tue， 如 何 使 用 站 语句 执行 代码 。 

当 一 条 让 语句 的 测试 表达 式 计算 为 false 时 ， 如 何 执行 供 选 择 的 代码 
块 。 

如 何 使 用 switch 语 句 根 据 测 试 表达 式 返 回 的 值 来 执行 代码 。 

如 何 使 用 while 语 句 来 重复 执行 代码 。 

如 何 使 用 for 语 句 使 循环 更 简洁 。 

如 何 跳出 循环 。 

如 何 将 一 个 循环 和 通 套 到 另 一 个 循环 中 。 

如 何在 控制 结构 中 使 用 PHP 的 start 和 end 标 签 。 


前 面 的 革 忆 中 创建 的 脚本 只 是 党 看 一 个 方 回 顺序 执行 。 也 束 是 说 ， 
每 次 脚本 运行 的 时 候 ， 同 样 的 语句 按照 同样 的 顺序 在 执行 。 这 样 就 没有 
多 少 录 活性 ， 因 为 各 种 动态 程序 设计 部 至少 需要 一 个 或 两 个 循环 ， 更 不 
必 说 在 顺序 执行 之 前 检查 未 些 条 件 的 正确 性 的 能 力 。 现 在 我 们 将 学 习 程 
序 设 计 结 构 ， 这 些 结构 使 得 脚本 适合 于 实际 环境 。 


6.1 转换 流程 


计算 条 件 并 因此 改变 上 自己 的 操作 ， 这 对 于 脚本 来 说 是 很 闸 见 的 。 这 
些 判断 使 PHP 页 面 变 得 动态 一 一 也 束 是 说 ， 能 够 根据 情况 改变 输出 。 和 
大 多 数 程序 设计 语言 一 样 ，PHP 人 允许 你 用 证 语句 来 做 到 这 一 点 。 





6.1.1 iZ 


计 语 名 是 一 种 控制 跟 在 其 后 面 的 语 铝 《 即 一 个 单个 的 语句 或 在 括 扎 
内 的 代码 块 ) 执行 的 方法 。 计 语句 计算 括号 之 内 的 表达 云 一 一 如 朱 这 个 
表达 式 结 来 为 true， 执 行 后 续 语 名， 全 则 完全 跳 过 语句 。 这 个 功能 允许 
脚本 根据 多 种 因 系 来 做 判断 。 


if (expression) 1 
// code to execute if the expression evaluates to true 


} 
只 有 当 变量 包含 字符 串 *happy" 时 ， 程 序 清单 6.1 才 执行 代码 块 。 
程序 清单 6.1 ”if 语句 
1: <?php 
2: $mood = "happy"; 
3: if ($mood == "happy") { 
4: echo "Hooray! I'm in a good mood!"; 
a. f} 
6: ?> 


在 第 2 行 ， 把 “happy” 赋 给 变量 $mood。 在 第 3 行 ， 比 较 运 算 符 == 把 
变量 $mood 的 值 与 字符 串 “happy” 做 比较 。 如 果 它 们 匹配 ， 表 达 式 计算 为 
true， 接 着 执行 后 续 代 人 码 ， 直 到 过 到 结束 的 花 括 号 《在 本 例 中 的 第 5 
mE 


把 这 些 代 人 码 放 到 名 为 testif.php 的 文本 文件 中 ， 并 把 文件 放 到 Web 服 
器 文档 根 日 录 下 。 当 通过 Web 浏 览 器 访问 这 个 脚本 时 ， 产 生 如 下 的 输 
“i 


Hooray! I'm in a good mood! 


如 果 你 把 $mood 的 值 改 为 “sad”* 或 者 除了 “happy” 以 外 的 其 他 字符 
串 ， 再 次 运行 这 个 脚本 ， 让 语句 中 的 表达 陈 将 计算 为 false， 导 致 忽略 后 
续 代 但 块 。 这 个 脚本 什么 都 不 做 ， 它 将 引导 我 们 到 else 子 句 。 


6.1.2 ”使 用 else 子 句 的 站 语句 


“AR AEA, PRA nt ence 如 果 测 试 
eA Tit He ANfalse, AA yi alanis WyGelsevs 0 4) a AER a 
其 他 代码 块 的 让 语 句 中 ， 可 以 实现 这 一 点 


if (expression) { 

// code to execute if the expression evaluates to true 
+ else { 

// code to execute in all other cases 


} 


程序 清单 6.2 完 善 了 程序 清单 6.1 中 的 示例 ， 因 此 ， 如 果 $mood 的 值 
不 守 于 “happy” 将 执行 一 个 默认 代码 块 。 


程序 清单 6.2 ”使 用 else 的 if 语 句 


1: <?php 

2: $mood = "sad"; 

3: if ($mood == "happy") { 

4: echo "Hooray! I'm in a good mood!"; 
5: } else { 

6: echo "I'm in a $mood mood."; 

PE f 

B: 2> 


PEX HE RAS INE 4 Atestifelse.phpiy <A LF, FERC ECE Web 
服务 器 文档 根 目 录 下 。 当 使 用 Web 浏 览 器 访问 这 些 脚 本 时 ， 产 生 如 下 的 
输出 。 

I'm in a sad mood. 


注意 ， 在 第 2 行 ，$mood 被 赋予 的 值 是 字符 串 “sad”， 显 然 不 等 
于 “happy”， 所 以 第 3 行 的 许 语句 的 表达 式 计 算 为 false。 这 导致 跳 过 了 第 
一 个 代码 块 (第 4 行 )， 而 执行 else 后 面 的 代码 块 并 显示 男 一 个 供 选择 的 
信息 : Pm in a sad mood。 字 符 串 “sad” 是 赋 给 变量 $mood 的 值 。 


使 用 一 个 与 ff 语句 联合 的 else 子 匀 ， 人 允许 脚本 进行 有 天 代 人 码 执行 的 判 
断 。 然 而 ， 这 种 选择 受 限 于 either-or 分 支 : 要 么 执行 跟 在 if 语 句 后 面 的 代 
码 块 ， 要 么 执行 跟 在 else 语 句 后 面 的 代码 块 。 现 在 我 们 将 学 习 用 于 计算 
多 个 表达 式 的 男 一 种 选择 ， 这 些 表 达 式 是 一 个 接 一 个 的 。 


6.1.3 ”使 用 带 有 elseif 子 句 的 让 语句 


提供 一 个 默认 代码 块 (elseif 部 分 ) 之 前 ， 你 可 以 使 用 一 个 if... 
elseif...else 子 句 来 测试 多 个 表达 式 (if...else 部 分 ) 。 


if (expression) 1 
// code to execute if the expression evaluates to true 
+ elseif (another expression) { 
// code to execute if the previous expression failed 
// and this one evaluates to true 
} else { 
// code to execute in all other cases 
} 


如 果 最 初 的 ff 表达 式 结 果 不 为 tue， 那 么 将 跳 过 第 一 个 代码 块 。 
elseif 子 句 给 出 男 一 个 表达 式 来 计算 ， 如 果 这 个 表达 式 计 算 为 true， 则 执 
行 它 对 应 的 代码 块 。 耕 则 ， 执 行 与 else 子 句 天 联 的 代码 块 。 你 可 以 包含 


fja 


提示 : 





elseif 子 句 也 可 以 写成 两 个 单词 else if) 。 采 用 哪 种 语法 是 个 人 喜好 问题 ， 但 是 
PEAR (PHP extension and application repository，PHP 扩 展 与 应 用 库 ) 和 PECL (PHP extension 
community library，PHP 扩 展 公 用 库 ) 所 米 用 的 编码 标准 都 使 用 elseif。 


程序 清单 6.3 在 前 一 个 示例 中 添加 了 elseif 子 句 。 


程序 清单 6.3 ”使 用 else 和 elseif 的 证 语句 


1% S7pnp 

2: $mood = "sad"; 

3: if ($mood == "happy") { 

4: echo "Hooray! I'm in a good mood!"; 

5: } elseif ($mood == "sad") { 

6: echo "Awww. Don't be down!"; 

7: } else { 

8: echo "I'm neither happy nor sad, but $mood."; 
9: } 

10: ?> 


把 $mood 变 量 再 次 赋值 为 “sad”， 如 第 2 行 所 示 。 因 为 这 个 值 
与 人 happy” 不 相等 ， 所 以 第 4 行 的 代码 被 跳 过 。 第 5 行 的 elseif 子 句 检 测 
$mood 有 的 值 与 “sad” 是 人 耕 相 等 ， 在 这 个 示例 中 结 来 为 tue， 因 此 执行 第 6 行 
的 代码 。 从 第 7 行 到 第 9 行 ， 提 供 了 一 个 默认 操作 ， 如 末 前 面 的 测试 条 件 
全 都 是 false， 那 么 将 调用 这 个 操作 。 在 这 个 示例 中 ， 我 们 只 简单 地 显示 
一 条 消 上 县， 其 中 包 舍 $mood 变 量 的 实际 值 。 


把 上 述 代码 放 到 名 为 testifelseif.php 的 文本 文件 中 ， 并 把 文件 放 到 
Web 服 务 器 文档 根 目 录 下 。 当 通过 Web 浏 览 器 访问 这 个 脚本 时 ， 产 生 如 


下 的 输出 。 
Awww. Don't be down! 
将 $mood 的 值 改 为 “iffy” 并 运行 这 个 脚本 ， 将 产生 如 下 的 输入 。 


I'm neither happy nor sad, but iffy. 


6.1.4 switchi= 4) 


switchif J se A- PHRI SIA SU 2G ROEM IE. EURE 
经 看 到 的 ， 使 用 让 和 elseif 语 句 可 以 计算 多 个 表达 式 。 然 而 ，switch 语 名 
只 计算 一 个 表达 陈列 表 ， 基 于 匹配 代 但 的 一 个 特定 位 来 选择 正确 的 一 
个 。 作 为 让 语句 一 部 分 的 表达 陈 的 结 采 要 么 是 true 要 么 是 false， 而 switch 
语句 的 表达 式 部 分 则 是 连续 检测 很 多 值 ， 布 望 找 到 一 个 匹配 的 值 。 


switch {expression} { 

case result?: 
// execute this if expression results in resulti 
break; 

case result2: 
// execute this if expression results in result2 
break; 

default: 
// execute this if no break statement 
// has been encountered hitherto 


switch 语 句 中 的 表达 式 通 党 只 是 一 个 变量 ， 例 如 $mood。 在 switch 语 
名] 中， 你 会 发 现 很 多 的 条 件 语句 。 每 一 个 条 件 都 检测 一 个 值 是 否 与 
switch 表 达 式 的 值 四 配 。 如 果 条 件 值 等 于 表达 式 的 值 ， 执 行 条 件 语句 中 
的 代码 。 最 后 ，break 语 句 将 结束 switch 语 句 的 执行 。 


如 末 省 略 了 break 语 名 ， 那 么 将 顺序 执行 下 一 个 条 件 语 句 ， 而 不 管 
前 一 个 匹配 的 值 是 任 已 经 找到 。 如 果 在 执行 到 可 选 的 献 认 语 句 之 前 还 没 


有 找到 一 个 匹配 的 值 ， 那 么 执行 默认 语句 。 
在 作为 条 件 语句 的 一 部 分 执行 的 任何 代码 的 最 后 ， 包 含 一 条 break 语 句 ， 这 是 很 重要 的 。 


没有 break 语 句 ， 程 序 流 程 将 继续 下 一 个 条 件 语 多， 并 最 终 到 达 黑 认 语句 。 在 大 多 数 情况 下 ， 
这 将 导致 一 个 无 法 预计 的 结 末 ， 而 这 个 结果 很 可 能 是 错误 的 ! 


程序 清单 6.4 使 用 switch 语 名 重新 创建 了 让 语 名 示例 的 功能 。 


程序 清单 6.4。” switch 语句 


1: <?php 

2: $mood = "sad"; 

3: switch ($mood) { 

4 case "happy": 

he echo "Hooray! I'm in a good mood!"; 
6 break ; 

7 case "sad": 

8: echo "Awww. Don't be down!"; 

9: break; 

10: default: 

i echo "I'm neither happy nor sad, but $mood."; 
12: break; 

Taz at 

14: T> 


在 第 2 行 把 $mood 变 量 的 值 初 始 化 为 “sad”。 第 3 行 的 Switch 语句 将 这 
个 变量 作为 它 的 表达 式 。 第 4 行 的 第 一 个 条 件 语句 检 负 “happy” 与 变量 
$mood 的 值 是 含 相 等 。 在 本 示例 中 两 着 并 不 匹配 ， 所 以 脚本 执行 到 第 7 
行 的 第 二 个 条 件 语句 。 字 从 串 “sad” 与 $mood 变 量 的 值 相等 ， 所 以 执行 这 
个 代码 块 。 第 9 行 的 break 语 句 结 束 这 个 过 程 。 第 10 行 到 第 12 行 提供 了 一 
个 默认 操作 ， 当 前 面 的 条 件 结果 没有 一 个 为 true 时 执行 它 。 


把 上 述 代 码 放 到 名 为 testswitch.php 的 文本 文件 中 ， 并 把 文件 放 到 


Web 服 务 器 文档 根 目 录 下 。 当 用 Web 浏 览 器 访问 这 个 脚本 时 ， 产 生 如 下 
的 输出 。 
Awww. Don't be down! 

把 $mood 的 值 改 为 “happy” 并 运行 脚本 ， 将 产生 如 下 的 输出 。 
Hooray! I'm in a good mood! 

为 了 强调 break 语 句 的 重要 性 ， 答 试 运行 疫 有 第 二 个 break 语 句 的 这 
个 脚本 。 确 保 把 $mood 的 值 改 回 为 “sad” 并 且 运 行 该 脚本 ， 输 出 如 下 。 
Awww. Don't be down!I'm neither happy nor sad, but sad. 

IMA Ee BTA! mh, a DAT FE ERS SA aA break 
1 HJ. 
6.1.5 EH? 运算 从 

? RZE SINEK, RATER EHME, 2X-“MEVEA 
使 用 冒号 分 隅 开 的 两 个 表达 式 中 的 一 个 。 这 个 结构 为 你 提供 了 作为 一 个 
整体 的 三 部 分 ， 因 此 得 名 三 元 操作 人 符 。 表 达 陈 通 弟 根据 测试 表达 式 的 结 
采 来 产生 返回 什 。 
(expression) ? returned if expression is true : returned if expression is false; 

如 果 测 试 表达 式 计 算 为 rue， 人 返回 第 二 个 表达 式 的 值 ， 耕 则 返回 第 
三 个 表达 式 的 值 。 程 序 清单 6.5 使 用 三 元 操作 符 根 据 $mood 的 值 设置 变量 
的 值 。 


程序 清单 6.5 使用? 运算 符 


一 上 


: <?php 


2: $mood = "sad"; 

3: $text = ($mood == "happy"}) ? "I am in a good mood!" : "I am in a $mood 
mood." ; 

4: echo "$text"; 

a. oo 


在 第 2 行 ， 把 $mood 设 为 “sad”。 在 第 3 行 ， 测 试 $mood 与 字符 
串 “happy” 是 否 相 等 。 因 为 测试 返回 false， 所 以 返回 三 元 操作 符 的 第 三 
个 表达 式 的 结果 。 


把 上 述 代 但 放 到 名 为 testten.php 的 文本 文件 中 ， 并 把 文件 牙 到 Web 
服务 颖 文档 根 目录 下 。 当 用 Web 浏 贤 器 访问 这 个 脚本 时 ， 产 生 如 下 和 输 
ae 


I am in a Sad mood, 


STUPRYE TT AR IEE fE, (Ee, MRP IR AP BE DY Sct PEE AG H tid vit A 
代码 来 完成 ， 它 惑 派 上 用 场 了 。 


6.2 ”循环 


迄今 为 止 ， 我 们 已 经 看 到 了 脚本 可 以 做 出 和 执行 什么 代码 相关 的 判 
汤 。 脚 本 还 可 以 判定 执行 一 个 代码 块 多 少 次 。 循 环 语句 专门 设计 来 允许 
你 完成 午 复 任务 ， 因 为 它们 继续 操作 下 到 达到 一 个 指定 条 件 或 耳 到 喧 式 
地 选择 退出 循环 。 


6.2.1 while 语句 


while 语 句 看 上 去 和 一 个 基本 的 if 语 句 结构 类 似 ， 但 它 有 循环 的 能 
力 。 


while (expression) 1 
// do something 
f 


和 让 语句 不 同 ，while 语 句 只 要 在 表达 式 结 束 为 true 的 情况 下 残 执 
行 ， 即 如 采 需 要 会 一 再 地 执行 。 循 环 中 代 但 其 的 每 次 执行 都 叫做 一 次 迭 
代 (iteration〉。 在 代码 块 中 ， 通 常会 改变 菏 个 影 啊 while 语 句 的 表达 式 
的 值 ; 否则 ， 循 环 将 永远 继续 。 例 如 ， 可 以 用 一 个 变量 来 计算 迭代 的 侈 
数 ， 并 据 此 采取 相应 的 操作 。 程 序 清单 6.6 创 建 一 个 while 循 环 ， 计 算 并 
显示 2 的 倍数 百 到 24。 


程序 清单 6.6” while 语句 


: <?php 

: $counter = 1; 

: while {$counter <= 12) { 

echo $counter." times 2 is ".($counter * 2)."<br />"; 
$countertt; 


} 


?> 


NOOR WDM — 


在 这 个 示例 中 ， 第 2 行 和 初始化 变量 $counter， 将 其 值 设 置 为 1。 第 3 行 
的 while 语 句 检 验 $counter 变 量 ， 在 $counter 的 值 小 于 或 等 于 12 时 ， 循 环 
将 继续 运行 。 在 while 语 名 的 代码 块 中 ，$counter 的 值 乘 以 2， 并 且 把 续 
果 显 示 到 浏览 器 中 。 在 第 5 行 ，$counter 的 值 每 次 增加 1。 这 一 步 非常 重 
要 ， 因 为 如 果 不 增 加 变量 $counter 的 值 ，while 表 达 式 将 决 不 会 转变 为 
false， 循 环 将 永 不 俘 止 。 


把 上 述 代 码 放 到 名 为 testwhile.php 的 文本 文件 中 ， 并 把 文件 放 到 
Web 服 务 器 文档 根 目录 下 。 当 用 Web 浏 览 器 访问 这 个 脚本 时 ， 产 生 如 下 
的 输出 。 


1 times 2 1S 2 
2 times 2 is 4 
3 times 2 is 6 
4 times 2 is 8 
5 times 2 is 10 
6 times 2 is 12 
7 times 2 is 14 


8 times 2 is 16 
9 times 2 1s 18 
1@ times 2 is 26 
11 times 2 18 22 
12 times 2 15 24 


6.2.2 ”do...while 语 名 


do...while 语 名 看 起 来 有 点 像 while 语 名 在 开始 的 地 方 打开 了 开关 。 
两 者 之 间 的 主要 差别 是 do...while 语 句 在 表达 式 真 值 检 测 之 前 执行 代码 
块 ， 而 不 是 在 表达 式 真 值 检测 之 后 执行 代码 块 。 
do { 


// code to be executed 
+ while (expression); 





do...while 语 句 的 测试 表达 式 应 该 始终 以 一 个 分 号 结束 。 


当 我 们 希望 代码 块 至 少 执行 一 次 时 ， 即 便 是 while 表 达 式 计算 为 false 
的 时 候 也 如 些 ，do...while 语 句 就 很 有 用 。 程 序 清单 6.7 创 建 了 一 个 do... 
while 语 句 。 这 个 代码 块 至 少 执行 一 次 。 


程序 清单 6.7 do...while 语 句 


: <?php 
: $num = 1; 
do { 
echo "The number is: ".$num."<br />"; 
$numt+; 
: } while (($num > 200) && ($num < 40@)); 
?> 


NOOR WN 一 


do...whilei# 4) tx Wl $num4e 2 4 ATK F200 87) F400 
值 。 在 第 2 行 ， 我 们 把 $num 人 初始 化 为 1L， 所 以 这 个 表达 式 的 结果 为 false。 
但 是 ， 在 计算 表达 式 之 前 至 少 执行 代 但 换 一 次 ， 所 以 do...while 语 句 在 
浏览 硕 中 显示 一 个 单独 的 行 。 

把 上 述 代 码 放 到 名 为 testdowhile.php 的 文本 文件 中 ， 并 把 文件 放 到 
Web 服 务 器 文档 根 目 录 下 。 当 用 Web 浏 览 器 访问 这 个 脚本 时 ， 产 生 如 下 
的 输出 。 

The number is: 1 

如 果 在 第 2 行 把 $num 的 值 改 为 像 300 这 样 的 值 并 运行 脚本 ， 循 环 显 

示 如 下 。 


The number 1s: 300 


并 日 将 随 着 数字 递增 继续 显示 类 似 的 行 ， 直 到 显示 如 下 。 
The number is: 399 


6.2.3 for 语句 


想 用 for 语 句 做 的 任何 事情 也 可 以 用 while 语 句 来 实现 ， 但 是 for 语 名 
通 间 是 达到 同样 效 来 的 更 局 效 的 方 读 。 在 程序 清单 6.6 中 ， 我 们 看 到 了 
如 何在 while 语 句 之 外 初始 化 变量 ， 接 着 检测 while 语 名 的 表达 式 并 在 后 
续 代 但 块 中 递增 变量 。for 语 句 也 可 以 做 这 样 的 一 系列 事情 ， 而 且 就 在 一 
个 单个 的 代码 行 中 。 这 使 代码 更 简洁 ， 并 且 较 少 有 生 类 似 由 于 瑟 记 递增 
变量 计数 需 而 导致 一 个 无 限 循环 的 情况 。 
for (initialization expression; test expression; modification expression) { 

// code to be executed 


} 


提示 : 


无 限 循环 ， 顾 名 思 义 ， 是 没有 限制 地 运行 的 循环 。 如 果 循环 正在 无 限 运行 ， 脚 本 将 运行 
无 限 长 时 间 。 这 种 行为 对 Web 服 务 器 造成 了 很 大 的 压力 并 且 导致 网 页 不 可 用 。 


for 语 句 的 圆 括 写 内 的 表达 式 用 分 与 分 隅 开 。 通 第 ， 第 一 个 表达 式 们 
始 化 一 个 计数 器 变量 ， 第 二 个 表达 式 是 循环 的 检测 条 件 ， 第 三 个 表达 式 
递增 计数 万 变量 。 程 序 清单 6.8 展 示 了 重新 创建 程序 清单 6.6 中 的 示例 的 
for 语 句 ， 这 个 示例 是 用 2 乘 以 12 个 数字 。 


程序 清单 6.8 ”使 用 for 语 句 


: <?php 
: for ($counter=1; $counter<=12; $counter++) { 
echo $counter." times 2 is ".{$counter * 2)."<br />"; 


} 


?> 


O eO N| 


把 上 述 代码 放 到 和 名 为 testfor.php 的 文本 文件 中 ， 并 把 文件 放 到 Web 服 
务 器 文档 根 目 录 下 。 当 用 Web 浏 览 器 访问 这 个 脚本 时 ， 产 生 如 下 的 输 


Ae 

1 times 2 is 2 
2 times 2 is 4 
3 times 2 is 6 
4 times 2 is 8 
5 times 2 is 1@ 
6 times 2 is 12 
7 times 2 is 14 
8 times 2 is 16 
9 times 2 is 18 


18 times 2 is 20 
11 times 2 18 22 
12 times 2 18 24 


程序 清单 6.6 和 6.8 的 结 采 完全 一 样 ， 但 是 for 语 句 使 得 程序 清单 6.8 中 
的 代码 更 加 人 窗 洁 。 因 为 在 语句 一 开始 束 初 始 化 $counter 变 量 并 递增 ， 循 
环 的 逻辑 一 目 了 然 。 也 就 是 ， 正 如 第 2 行 看 到 的 ， 第 一 个 表达 式 初 始 化 
$counter 变 量 并 赋值 为 1， 测 斌 表达 陈 验证 $counter 包 含 一 个 小 于 或 等 于 
12 的 值 ， 并 且 最 后 的 表达 式 使 counter 变 量 递增 。 这 些 表 达 式 中 的 每 一 
个 都 包含 在 代码 的 单独 一 行 中 。 


当 脚 本 执行 的 顺序 到 达 for 循 环 时 ， 初 始 化 $counter 变 量 并 计算 测试 
KISIN WR RIAA Z seinen MÍT. FG HT S$countere 
量 并 再 次 计算 测试 表达 式 。 这 个 过 程 一 下 持续 到 测试 表达 式 的 结果 为 


false. 


6.2.4 ”用 break 语 句 跳出 循环 


while 语 句 和 for 语 句 痢 包含 了 一 个 可 以 用 来 结束 循环 的 内 建 表达 
式 。 然 而 ，break 语 名 允许 你 根据 补充 测试 的 结果 来 跳出 循环， 这 为 预 
防 销 误 提供 了 一 种 保护 措施 。 程 序 清 单 6.9 创 建 了 一 个 简单 的 for 语 句 ， 
它 用 一 个 很 大 的 数 除 以 一 个 速 增 的 变量 ， 在 屏 大 上 输出 结果 。 


程序 清单 6.9 ”用 束 增 到 10 的 数 来 除 4000 的 for 循 环 


1: <?php 

2: for ($counter=1; $counter <= 10; $countert+t+) { 

at $temp = 4000/$counter; 

4: echo "4000 divided by ".$counter." is...'.$temp."<br />"; 
ae F 

G: Ta 


在 第 2 行 ， 这 个 示例 初始 化 $counter 变 量 并 赋值 为 1。for 语 句 中 的 测 
试 表 达 式 验证 $counter 的 值 小 于 或 等 于 10。 在 代码 块 中 ，4000 除 以 
$counter, FFI Ze ANSI as o 


EERE EJZ Atestfor2.php ty CA LFF, FPF SCE El] Web 
IRA as CPR A Se RK. “SX Webixll ids Wy Tal PS IASI, PAE GOB ASA 
HH o 


4000 divided by 
4000 divided by 
4000 divided by TS 1333,.33333333 
4000 divided by is... 1000 


1 1S... 4000 
2 
3 
4 

4000 divided by 5 is... 800 
6 
7 
8 
9 
1 


is... 2000 


4000 divided by is... 666.666666667 
4000 divided by is... 571.428571429 
4000 divided by is... 500 

4000 divided by is... 444,444444444 
4000 divided by 10 is... 400 


WF CA Ae 了。 但 是 如 来 你 用 入 到 $counter 的 值 来 自用 户 的 
得 入 呢 ? 这 个 信 可 能 旦 一 个 负数 ， 其 全 在 一 个 字符 串 。 让 我 们 回头 看 第 
一 个 示例 ， 用 户 输 入 了 一 个 负数 。 把 $counter 的 初始 值 从 1 改变 为 -4， 妆 


第 5 次 执行 代码 块 时 将 发 生 4000 除 以 0。 对 于 你 的 程序 来 说 ， 被 0 除 通 各 
不 是 一 个 好 主意 ， 因 为 这 样 的 操作 导致 的 结果 是 “undefined”。 如 果 变 量 
$counter 的 值 等 于 零 ， 程 序 清单 6.10 通 过 跳出 循环 防止 了 这 种 情况 的 发 
Æ, 


程序 清单 6.10 “使 用 break 语 名 


1: <?php 

2: $counter = -4; 

3: for (; $counter <= 10; $countert++) { 
4: if ($counter == 0) { 

5: break; 

6: } else { 

T: $temp = 4000/$counter; 

8: echo "4000 divided by ".$counter." is...".$temp."<br />"; 
9: } 

10: } 

11 ?> 





在 PHP 里 用 一 个 数 除 以 零 不 会 导致 一 个 致命 的 错误 。 相 反 ，PHP 产 生 一 个 警告 并 继续 执 


如 第 4 行 有 所 示 ， 我 们 使 用 了 一 个 if 语 句 ， 试 图 在 数学 运算 中 使 用 这 个 
值 之 前 ， 检 测 $counter 的 值 。 如 果 $counter 的 值 等 于 零 ， 那 么 break 语 句 
立即 保 止 执行 代码 块 ， 并 量程 序 将 在 for 语 名 (第 11 行 ) 之 后 继续 执行 。 


把 上 述 代码 放 到 名 为 testfor3.php 的 文本 文件 中 ， 并 把 文件 放 人 到 Web 
服务 左 文 档 根 目录 下 。 当 用 Web 浏 览 右 访问 这 个 脚本 时 ， 产 生 如 下 的 输 
出 。 


4000 divided by -4 is... -1000 
4000 divided by -3 is... -1333.33333333 
4000 divided by -2 is... -2000 
4000 divided by -1 is... -4000 


注意 ， 变 量 $counter 在 第 2 行 初 始 化 ， 并 不 在 for 语 句 中 的 圆 括 扎 内。 
这 个 方法 通 音 用 来 模拟 一 种 情况 ， 其 中 $counter 的 值 在 脚本 之 外 设 定 。 


你 知道 吗 ? 





你 其 至 可 以 省 略 for 语 句 中 的 所 有 表达 式 ， 但 是 必须 保留 分 隅 的 分 气 。 


6.2.5 /Hcontinueit= a) Pitz ft 


continuel® ®) 2 R 4 AVIAARI EAT, (EAN BCE AAR. AH 
Re, RHA RMA. PERS 6.10 18 A breakif a) A A, 
FER Pela 26.114 (8 A continuei, AHE RIEA ait BY DAE ae BRD 
OM FT TR o 


程序 清单 6.11 使 用 continue 语 句 


<?php 
$counter = -4; 
for {; $counter <= 10; $counter++) { 
if ($counter == 0) { 
continue; 


$temp = 4000/$counter; 
echo "4000 divided by ".$counter." is...".$temp."<br />"; 


=- OOnN OO 上 hm 一 


第 5 行 ， 我 们 用 continue 语 句 符 换 了 了 break 语句 。 如 果 变 量 $counter 的 
值 等 于 0， 跳 过 当前 迭代 并 立即 开始 下 一 次 迭代 。 


把 上 述 代 码 放 到 名 为 testcontinue.php 的 文本 文件 中 ， 并 把 文件 放 到 


Web 服 务 器 文档 根 目 录 下 。 当 用 Web 浏 览 右 访问 这 个 脚本 时 ， 产 生 如 下 
的 输出 。 


4000 divided by -4 is... -1000 

4000 divided by -3 is... -1333.33333333 
4000 divided by -2 is... -2000 

4000 divided by -1 is... -4000 

4000 divided by 1 is... 4000 

4000 divided by 2 is... 2000 

4000 divided by 3 is... 1333.33333333 
4000 divided by 4 is... 1000 

4000 divided by 5 is... 800 

4000 divided by 6 is... 666.666666667 
4000 divided by 7 is... 571.428571429 
4000 divided by 8 is... 500 

4000 divided by 9 is... 444.44444444444 
4000 divided by 10 is... 400 


po 
VERS: 


使 用 break 和 continue 语 句 将 让 代码 变 得 更 难以 阅读 ， 因 为 这 增加 了 包含 它们 的 循环 语句 的 
逻辑 层次 的 复杂 性 。 小 心地 使 用 这 些 语句 ， 或 者 对 展示 给 其 他 程序 员 (或 者 你 自己) 的 代码 
添加 注释 ， 说 消 苞 你 使 用 这 些 语句 所 要 达到 的 目的 。 








6.2.6 KEJA 


HAEA Ae A Aa a), AEA ANHEE. Sa) 
AS EN GEHTML AY, EA A AZ a RIA A. Pe APY 6.12 (5 FA 
“forte J £E ill i ae FL ZN PS FRE o 


程序 清单 6.12 REMS fori 


<?php 
echo "<table style=\"border: 1px solid #@@@;\"> \n"; 
for ($y=1; $y<=12; $y++) { 
echo “<tr> Wr 
for ($x=1; $x<=12; $x++) { 
echo "<td style=\"border: 1px solid #000; width: 25px; 
text-align:center;\">"; 


NAO BW Nh — 


8: echo ($x * $y); 
9: echo "</td> \n'; 


5 0 E- acho “<FEr= yn- 


} 
13: ecno “</table>; 
14: ?> 


FE Ke Sr for Tea, LEB aE aE 6.12" A BB 247 


echo "<table style=\"border: 1px solid black;\"> \n"; 


注意 ， 在 包含 表格 样式 信息 的 字符 串 中 的 每 个 引号 之 前 ， 我 们 使 用 
TRAL O 。 这 些 反 冬 杠 也 出 现在 第 6 行 和 第 7 行 中 的 表格 数据 单元 
的 样式 信息 中 。 这 十 必要 的 ， 因 为 它 告 诉 PHP3 引 敬 我 们 想 要 使 用 引号 字 
伯 ， 而 个 是 让 PHP 把 它 解 释 成 字符 串 的 开始 或 结束 。 如 来 没有 用 肥 冬 杠 
子 稚 将 引号 转 义 ， 访 语句 对 于 引擎 来 说 就 没有 意义 了 ， 因 为 引 黎 将 把 语 
名 理解 为 一 个 字符 里 后 面 跟 看 一 个 数 子 ， 数 子 后 面 再 跟 看 力 一 个 字符 
囊 。 这 样 一 个 结构 可 能 产生 错误 。 也 是 在 这 行 ， 我 们 使 用 \n 表 示 换 行 字 
伯 ， 一 旦 将 它 提交 给 浏览 荔 ， 源 程序 将 更 容易 理解 。 


外 层 的 for 语 句 〈 第 3 行 ) 初始 化 一 个 名 为 $y 的 变量 ， 并 为 它 赋 予 初 
始 值 1。 这 个 for 语 句 定义 一 个 准备 验证 $y 的 值 小 于 或 等 于 12 的 表达 式 ， 
接 看 定义 一 个 将 要 用 到 的 增 量 。 在 每 次 从 代 中 ， 代 人 码 块 输出 一 个 tt CR 
格 行 ) 的 HTML 元 系 〈 第 4 行 ) 并 开始 内 层 for 语 句 〈 第 5 行 )。 这 个 内 层 
循环 初始 化 一 个 名 为 $x 的 变量 ， 并 且 定 义 了 和 外 循环 一 致 的 表达 式 。 对 
于 每 次 达 代 ， 内 循环 输出 一 个 td GRR Hc) TORR BN bias (ROT 
和 第 7 行 )， 还 输出 $x 乘 以 $y 的 结果 (第 8 行 )。 在 第 9 行 ， 我 们 关闭 表 
格 单 元 格 。 内 层 的 循环 完成 后 ， 我 们 退回 到 外 面 的 循环 ， 在 第 11 行 结 
表格 行 ， 准 备 再 次 开始 这 个 过 程 。 当 外 层 循 环 完 成 时 ， 显 示 一 个 整齐 的 
带 格 式 的 乘法 表 。 我 们 在 第 13 行 通过 结束 表 来 把 所 有 表格 信息 包 起 来 。 


把 上 述 代码 放 到 名 为 testnestfor.php 的 文本 文件 中 ， 并 把 文件 放 到 
Web 服 务 器 文档 根 目 录 下 。 当 用 Web 浏 览 器 访问 这 个 脚本 时 ， 产 生 如 图 
6-1 所 示 的 输出 。 
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图 6-1 testnestfor.php 的 输出 


6.3 {UAE AT bt a A OO 


在 第 4 章 ， 你 已 经 学 习 了 使 用 PHP 开 始 和 结束 标记 任意 切换 HTML 格 
式 。 在 本 间 中 ， 你 已 经 发 现 ， 根 据 用 过 和 switch 语 句 控 制 的 判断 处 理 过 
程 ， 我 们 可 以 对 用 户 呈 现 不 同 的 输出 。 在 本 节 ， 我 们 将 结合 这 两 种 技 
ye 


想象 一 个 脚本 ， 只 有 妆 把 一 个 变量 设 为 布尔 值 true 时 ， 它 才 输 出 一 
个 表格 的 值 。 程 序 清单 6.13 展 示 了 用 if 语 句 的 代码 块 构 建 的 一 个 简化 的 
HTML 表 格 。 


程序 清单 6.13 ”包含 多 个 echo 语 句 的 代码 块 


: <?php 
: $display prices = true; 
if ($display prices) { 
echo "<table border=\"1\">\n"; 
echo "<tr><td colspan=\"3\">"; 
echo "today's prices in dollars"; 
echo "</td></tr>"; 
echo "<tr><td>\$14.00</td><td>\$32.00</td><td>\$71.00</td></tr>\n"; 
echo "</table>"; 


—- =$- OA N OO BRWAND — 
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来 转 义 所 有 用 于 HTML 输 出 的 引号 。 


把 上 述 代 码 放 到 名 为 testmultiecho.php 的 文本 文件 中 ， 并 把 文件 放 到 


Webfk 4 a8 CMIR ASF. 4 Webi bi asv XAAR, PAE I 
6-2 所 示 的 输出 。 





图 6-2 ”testmultiecho.php 的 输出 


用 这 种 编码 方法 没有 任何 错 ， 但 通过 在 代码 块 中 回 到 HTML 模 式 ， 
我 们 瓯 可 以 省 去 茶 些 输入 。 在 程序 清单 6.14 中 我 们 束 是 这 样 做 的 。 


程序 清单 6.14 返回 代码 块 中 的 HTML 模式 


: <?php 

: $display prices = true; 

if ($display prices) { 

?> 

<table border= 1 > 

<tr><td colspan="3">today's prices in dollars</td></tr> 
 <tr><td>$14.00</td><td>$32.00</td><td>$71.00</td></tr> 
</table> 


OAN OTA OMN 一 
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在 第 4 行 转换 到 HTML 模式 。 这 可 以 避免 我 们 要 转 义 引号 和 把 输出 族 入 
全 echo() 语 句 中 的 及 烦 。 然 而 ， 长 远 来 讲 ， 这 种 方法 可 能 影响 代码 的 可 
谈 性 ， 尤 其 是 在 脚本 变 得 越 来 越 大 的 时 候 。 


6.4 ”小 结 


在 本 章 中 ， 我 们 学 习 了 控制 结构 以 及 能 帮助 我 们 把 脚本 变 得 更 灵 
活 、 叶 动态 的 方法 。 这 些 结构 中 的 大 多 数 将 在 本 书后 续 部 分 中 经 第 出 
现 。 


我 们 学 习 了 如 何 定义 让 语句 以 及 如 何 用 elseif 和 else 子 句 提供 可 供 选 
择 的 操作 。 我 们 学 习 了 如 何 根据 对 一 个 表达 式 的 结 末 进行 多 次 相等 的 检 
测 ， 使 用 switch 语 句 改 变法 程 。 我 们 学 习 了 有 关 循 环 的 内 容 ， 特 别 是 
while 和 for 语 句 ， 还 学 习 了 如 何 使 用 break 和 continue 来 提前 结束 循环 的 执 
行 或 跳 过 一 次 欠 代 。 我 们 学 习 了 如 何 把 一 个 循环 众 矢 到 夯 一 个 循环 之 
中 ， 并 且 看 到 了 这 个 结构 的 一 个 典型 用 法 。 最 后 ， 我 们 看 到 了 一 种 使 用 
PHP 的 开始 和 结束 标记 与 条 件 代 码 块 协作 的 技术 ， 从 而 不 必 上 再 对 诸如 引 
写 和 美元 从 号 等 特殊 字符 转 义 《在 前 面 使 用 反 冬 杠 )。 


我 们 现在 应 该 学 会 了 尽 够 的 基础 来 目 己 编写 做 判断 和 执行 章 复 任务 
的 脚本 。 在 下 一 革 ， 我 们 将 看 到 问 应 用 程序 中 洪 加 更 多 能 力 的 一 种 新 方 
法 。 将 学 习 如 何在 函数 中 组 织 代 码 ， 以 防止 复制 和 提高 可 重用 性 。 


6.5 Q&A 


Q: 控制 结构 的 测试 表达 式 必 然 产 生 一 个 布尔 值 吗 ? 


A: ÆA EEN, (AED AIAN ERICH, FE. STAREX 
的 变量 或 者 一 个 空 字符 串 转 换 为 false。 所 有 的 其 他 值 将 计算 为 true。 


Q: 控制 语句 中 代码 块 必须 总 是 使 用 方 括 号 括 起 来 吗 ? 
A: 如 来 想 要 执行 的 代码 作 为 控制 结构 的 一 部 分 ， 且 这 个 控制 结构 


仅 由 一 个 单独 的 行 组 成 ， 那 么 可 以 省 略 方 括号 。 然 而 ， 不 宫 络 构 的 长 度 
有 多 长 ， 总 是 使 用 开始 和 结束 的 方 括号 古 一 个 好 习惯 。 


6.6 ”实践 练习 


实践 练习 是 设计 用 来 帮助 你 预料 可 能 的 问题 、 复 习 已 经学 过 的 知 
识 ， 并 且 开 始 把 知识 用 于 实践 。 


练习 题 


1. 如 果 一 个 整数 变量 $age 介 于 18 到 35 之 间 ， 你 将 如 何 使 用 站 语句 来 
[ay YX) a as an ch FB “Youth message”? 如 采 $age 包 含 任 何其 他 的 值 ， 字 


WA BĘ 


FFE “Generic message” 将 显示 在 浏览 器 上 。 


2. 如 果 变 量 $age 介 于 1 到 17 之 间 ， 你 将 如 何 扩展 问题 1 中 的 代码 以 
输出 字符 串 “Child message”? 


3. 如何 创建 一 个 在 1 到 49 之 间 递 增 并 输出 每 一 个 奇数 的 while 语 
句 ? 


4. 如何 将 问题 3 中 所 建立 的 while 语 名 转换 为 一 个 for 语 句 ? 


Fa 
oy 


— 


$age = 22; 


if (($age >= 18) && ($age <= 35)) { 
echo “Youth message"; 

} else { 
echo "Generic message"; 


} 


Gage = 12; 


if (($age >= 18) && (Sage <= 35)} 1 
echo "Youth message"; 

} elseif (($age >= 1) && {$age <= 17)) { 
echo "Ghild message"; 

} else { 
echo "Generic message"; 

} 


$num = 1; 


while ($num <= 49) { 
echo $num."<br > 
$num += 2; 


4. 


for {$num = 1; $num <= 49; $num += 2) { 
echo $num."<br /> 


} 


EA wel 
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2. 开始 构建 你 将 要 使 用 的 控制 结构 。 暂 时 使 用 临时 变量 来 模拟 用 
P $A Baa Fe A H « 


在 本 章 中 ， 你 将 学 到 : 


如 何在 脚本 中 定义 和 调用 水 数 。 

如 何 给 函数 传 素 值 ， 以 及 接受 由 此 返回 的 值 。 

如 何 使 用 存储 在 变量 中 的 一 个 子 从 里 来 动态 地 调用 一 个 函数 。 
如 何在 函数 中 访问 全 局 变量 。 

如 何 给 函数 一 个 “内 存 ”。 

如 何 通 过 引用 把 数据 传 圳 给 函数 。 

如 何在 调用 前 验证 一 个 函数 存在 。 


对 于 一 个 组 织 良 好 的 脚本 来 说 ， 函 数 是 其 核心 ， 并 且 函 数 使 得 代码 
BA AIA. WRIA PAL, KAI EH, BIN, EA 
性 代码 的 问题 将 会 使 开发 过 程 陷入 困境 。 在 本 章 中 ， 我 们 将 研究 函数 ， 
并 且 展 示 函 数 可 以 使 你 避免 重复 性 工作 的 一 些 ”方法 。 


7.1 o HR 


FAT AT CAFE eh BU TE ET A/S HH ALS © SSL SAS 
arhi 的 原材料 并 且 用 这 些 原材料 生产 出 一 个 产品 (输出 〉。 

图 数 接 党 值 并 处 理 它 们 ， 然 后 执行 一 个 操作 《〈 例 如， 显示 到 浏览 
Magarin 回 一 个 新 值 ， 或 者 既 执 行 操作 又 返回 值 。 


如 末 你 需要 烘焙 一 个 香料 ， 你 可 能 目 己 做 它 ， 在 目 己 的 厨房 里 使 用 
标准 燃 炉 。 但 是 ， 如 你 需要 供 烧 上 王 个 香 糙 ， 可 能 需要 制造 或 购买 一 
台 专 门 的 捍 糙 烘焙 机 ， 从 而 保证 大 量 地 烘焙 蛋糕 。 闫 似 地 ， 当 雇 定 是 合 
创建 一 个 函数 以 便 重 用 的 时 候 ， 需 要 考 夸 的 最 重要 的 因 系 是 它 可 以 节省 
你 编写 重复 性 代码 的 程度 。 


K% (function) 是 一 个 和 目 包含 的 代码 氛 ， 可 以 由 脚本 调用 。 当 调 
用 的 时 候 ， 束 执行 函数 的 代 但 来 完成 一 个 特定 的 任务 。 可 以 同一 个 函数 
传递 值 ， 函 数 束 会 适当 地 使 用 这 些 值 ， 和 存储 它们 、 改 变 它 们 、 显 示 它 
们 ， 或 者 做 任何 告诉 函数 要 去 做 的 事情 。 完 成 以 后 ， 函 效 也 可 以 同调 用 
它 的 最 初 的 代码 返回 一 个 值 。 


7.2 ”调用 函数 


图 数 分 为 两 闪 ， 内 建 于 语言 中 的 函数 和 目 己 定义 的 函数 。PHP 有 数 
百 个 内 建 的 函数 。 看 看 下 和 面 的 代码 给 出 的 使 用 函数 的 一 个 例子 。 


strtoupper{ "Hello Web!"}; 


这 个 例子 调用 了 strtoupper() 函 数 ， 把 字符 串 “Hello Web!” 传 递 给 
它 。 然 后 这 个 函数 负 贡 上 日 己 的 事务 ， 把 该 字 从 串 的 内 容 更 改 为 大 写字 
母 。 函 数 调 用 由 函数 名 在 这 个 例子 中 是 strtoupper〉 后 面 跟着 一 个 括号 
组 成 。 如 果 想 要 问 函 数 传递 信息 ， 可 以 把 它 放 到 括 写 之 间 。 按 照 这 种 方 
法 传递 给 函数 的 一 baw 昌 叫 做 参数 (argument) . Fete ARS hE 
PURER ECI KESAH E S ka RAA F 
some_function($an_argument, $another_argument); 

strtoupper() A ce — “PE eR, ALA ERA. KB BLA 
ME E A CONES Ze RRA, Ea 22> a A OWE 
务 是 否 成 功 执行 。strtoupper0 疯 数 返 回 一 个 字符 串 值 ， 因 此 它 的 用 法 逢 
要 使 用 一 个 变量 来 接受 这 个 新 的 字符 串 ， 示 例如 下 。 


$new string = strtoupper("Hello Web!"}; 


DLE, RIJA UER HEH Snew_string, WE bE EER E © 


echo $new string; 


TAT ARG EFS PA CAS NF RE 


HELLO WEB! 


提示 : 





print() 和 echo() 函 数 并 不 是 真正 的 冰 数 ， 它 们 是 用 来 设计 把 字符 串 输出 到 浏览 右 的 语言 构 
造 。 然 而 ， 你 会 在 PHP 的 函数 列表 中 找到 它们 ， 分 别 位 于 http:/www.php.net/print 和 
http://www.php.net/echo 。 这 些 构 造 在 功能 上 相似 并 且 可 以 互 换 地 使 用 。 使 用 哪个 只 是 你 的 个 
MN o 


例如 ，abs0O 函 数 需要 一 个 带 符号 的 数字 值 作为 参数 ， 并 且 返 回 该 数 
字 的 绝对 值 。 让 我 们 在 程序 清单 7.1 中 试 一 试 它 。 


程序 清单 7.1 调用 内 建 的 abs(0) 函 数 


: <?php 

: $num = -321; 

$newnum = abs($num); 
echo $newnum; 
Horints “321° 

?> 


中 AABN 


在 这 个 例子 中 ， 我 们 把 值 ~321 赋 给 一 个 变量 $num。 然 后 ， 我 们 把 
这 个 变量 传递 给 abs( 〇 函数， 该 函数 进行 必要 的 计算 并 返回 一 个 新 值 ， 我 
们 把 这 个 新 值 赋值 给 变量 $newnum， 人 然后 显示 结果 。 

把 上 述 代 码 放 入 到 一 个 名 为 abs.php 的 文本 文件 中 ， 并 将 该 文件 放置 
在 Web 服 务 器 文档 根 日 录 下 。 当 通过 Web 浏 览 器 来 访问 这 个 脚本 的 时 
修 ， 它 产生 如 下 结果 。 
321 

实际 上 上， 我们 完全 可 以 不 用 临时 变量 ， 而 是 把 我 们 的 数值 直接 传递 
2 abs() RŽ H A eS 


echo abs(-321}; 


然而 ， 我 们 还 是 使 用 了 临时 变量 $mum 和 $newnum， 从 而 使 这 个 过 
程 的 每 一 步 都 尽 可 能 的 清楚 。 有 了 时候， 我 们 可 以 通过 把 代码 分 解 成 很 多 
的 简单 表达 式 从 而 使 其 可 读 性 更 好 。 


可 以 控 照 和 我 们 调用 内 建 测 数 完全 相同 的 方式 来 调用 用 尸 定 义 的 函 
数 。 


7.3 ”定义 一 个 函数 


我 们 可 以 使 用 function 语 句 来 定义 目 己 的 函数 。 


function some function($argument1, $argument2) 


{ 


//function code here 


} 


图 数 的 名 字 跟 在 function 天 键 词 的 后 面 ， 并 且 后 面 有 一 对 括号 。 如 
朱 函 数 需 要 参数 ， 必 须 把 带 号 分 隔 开 的 变量 名 放置 在 括号 中 ， 这 些 变 量 
将 会 在 函 alia HIREA RAER. BUTERA BBB, thu 
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大 小 写 束 是 可 以 给 你 目 己 的 代码 添加 的 一 种 风格 ， 在 名字 中 使 用 混合 大 小 写 ， 例 如 
myFunction() 或 handleSomeDifficultTask()， 会 使 你 的 代码 更 容易 阅读 。 你 可 能 昕 说 过 这 种 叫做 
骆驼 命令 法 或 小 写 矣 弦 命 令 法 的 命令 规则 ， 二 者 的 产 别 在 于 第 一 个 字母 是 否 小 写 。 








程序 清单 7.2 声 明 并 调用 了 一 个 函数 。 
程序 清单 7.2 声明 并 调用 一 个 函数 


: <?php 
: Function bighellot) 


{ 
echo "<h1>HELLO!</h1>"; 


NOOR WDM = 


} 
: bighello(); 
?> 


程序 清单 7.2 中 的 脚本 输出 了 包含 在 一 个 HIML 的 h1 元 系 中 的 字符 
Ht “HELLO!” 


把 上 述 代 码 放 入 到 一 个 名 为 bighello.php 的 文本 文件 中 ， 并 且 将 这 
个 文件 放置 在 web 服务器 文档 根 目 孙 下 。 当 通过 Web 浏 览 右 访问 这 个 脚 
本 有 的 时 候 ， 应 访 看 到 如 图 7-1 所 示 的 结果 。 





" Fa] localhost/07/bighello.php 


€ C QO http://localhost/07/bighello.php ` 


HELLO! 


图 7-1 bighello.php 的 输出 


在 程序 清单 7.2 中 ， 我 们 声明 了 一 个 困 数 bighello0， 它 不 需要 参数 ， 
Al, Bias ree. RE SOE A, {LEFF 
不 是 非常 有 用 。 程 序 清单 7.3 创 建 了 一 个 水 数 ， 函数 需要 一 个 参 
数 ， 并 且 实 际 地 用 它 做 一 些 事情 。 


程序 清单 7.3 ”声明 需要 一 个 参数 的 阔 数 


: <?php 
: function printBR($txt) 


{ 
echo: SEKE Dr 


: printBR("This is a line."); 

>: printBR("This is a new line."); 

s printBR("This is yet another Line. =k; 
7a 


OAnN OoahAN 一 


提示 : 


和 变量 名 不 同 ， 函 数 名 是 不 区 分 大 小 写 的 。 在 前 面 的 例子 中 ，printBRO 函 数 也 可 以 叫做 
printbrO0、PRINTBR0 或 者 任意 大 小 写 组 合 ， 都 能 够 成 功 调用 该 函数 。 


把 上 述 代码 放 入 到 一 个 名 为 printbr.php 的 文本 文件 中 ， 并 且 将 这 个 
文件 放置 到 Web 服 务 右 文档 根 目 隶 下 。 当 通过 Web 浏 览 右 访问 这 个 脚本 
的 时 候 ， 应 诅 看 到 如 图 7-2 所 示 的 绪 末 。 


eS [| Ei 
/@ localhost/07/printbr. php 
é © | © http://localhost, 07/or ntbr.php a, | 
This is a line. | 
This is a new ine. | 
This is yet another ine. 


图 7-2” 显示 一 个 字符 串 以 及 一 个 附加 的 <br/> 标 记 的 函数 


在 第 2 行 ，printBRO 函 数 期 竺 一 个 字符 串 ， 因 此 ， 当 我 们 声明 该 函 
数 的 时 候 我 们 在 括号 中 放 入 一 个 变量 名 $txt。 不 各 传递 给 printBRO 的 是 
什么 ， 都 将 存储 到 $txt 变 量 中 。 在 这 个 国 数 体 中 ， 在 第 3 行 ， 我 们 刀 示 出 
$txt 变 量 ， 并 为 其 添加 一 个 <bw> 元 系 。 


当 我 们 想 要 辐 浏 席 硕 显示 一 行 的 时 候 ， 例 如 在 第 5 行 、 第 6 行 和 第 7 
行 ， 我 们 可 以 调用 printBRO 而 不 是 内 建 的 print()， 这 束 为 我 们 省 去 了 输 
A <br/> Jt HY RII -o 
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在 前 面 的 例子 中 ， 我 们 在 printBRO 函 数 中 向 浏览 器 输出 了 一 个 修改 
后 的 字符 串 。 然 而 ， 有 时候， 我 们 需要 一 个 函数 来 提供 一 个 可 供 目 己 使 
用 的 值 。 如 采 我 们 的 函数 已 经 修改 了 我 们 所 提供 的 一 个 字符 串 ， 我 们 可 
能 想 要 获得 修改 后 的 字符 串 ， 以 便 能 够 把 它 传递 给 其 他 的 函数 。 函 数 可 
以 使 用 retum 语 名 连接 一 个 值 ， 从 而 返回 一 个 值 。retum 语 句 停 止 了 当前 
图 数 的 执行 并 且 把 值 返回 给 调用 函数 的 代码 。 


程序 清单 7.4 创 建 了 一 个 返回 两 个 数字 的 和 的 函数 。 
程序 清单 7.4 返回 一 个 值 的 函数 


: <?php 
: function addNums($firstnum, $secondnum) 
i 
$result = $firstnum + $secondnun; 
return $result; 


} 
: echo addNums({3,5); 
=- /fwill print "8" 
?> 


OAnN ODOR 一 


把 上 述 代 码 放 入 到 一 个 名 为 addnums.php 的 文本 文件 中 ， 并 且 将 这 
个 文件 放置 在 Web 服 务 右 文档 根 目 录 下 。 当 通过 Web 浏 只 器 来 访问 这 个 
脚本 的 时 候 ， 它 产生 如 下 结 朵 。 

8 

注意 RO R E 
个 例子 中 ， 第 6 行 显示 这 两 个 参数 分 别 是 3 和 5) 。 这 些 值 存储 在 
$firstnum 和 $secondnum 弯 量 中 。 正 如 预期 的 那样 ，addNums0O 把 包 念 在 


这 些 变 量 中 的 数字 加 起 来 并 且 把 结果 存储 到 名 为 $result 的 变量 中 。 


return 语 人 句 可 以 返回 一 个 值 或 者 什么 也 不 返回 。 而 如 何 得 到 return 语 
句 所 返回 的 值 则 各 有 不 同 。 这 个 值 可 以 如 下 和 直接 编 但 。 


return 4; 


它 也 可 以 是 一 个 表达 却 的 结 东 ， 如 下 所 示 。 


return $a/$b; 


SA] he — TR BO FAR, SR TAN. 


return another function({$an argument}; 


7.5 Aber fe Ai 
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的 外 部 或 者 在 其 他 函数 中 是 不 可 用 的 。 在 较 大 的 项 目 中 ， 当 你 在 不 同 的 
图 数 中 声明 两 个 同名 的 变量 时 ， 这 样 做 会 防止 意外 地 上 履 兰 一 个 变量 的 内 


Y 
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之 外 显示 它 ae: 


程序 清单 7.5 ”变量 作用 域 : 在 一 个 函数 中 声明 的 变量 在 函数 外 是 不 可 用 的 


1: <?php 

2: function test() 

ois 

4: $testvariable = "this is a test variable"; 
D: 

6: echo "test variable: ".$testvariable."<br/>"; 

t 


?> 


把 上 述 代 码 放 入 到 一 个 名 为 scopetest.php 的 文本 文件 中 ， 并 且 将 这 
个 文件 放置 在 web 服务 喜 文 档 根 目 录 下 。 当 通过 web 浏览 器 来 访问 这 个 
脚本 的 时 候 ， 它 产生 如 图 7-3 所 示 的 结果 。 





| fa] localhost/O?/scopetest.php 


| > C | © htto://localhost/07/scopetest.ohp . | 


Notice: Undefined vanable: testvanable m C:\xampp\htdocs\07\scopetest.php 
on line 5 
test variable: 





图 7-3 ”scopetest.php 的 输出 
提示 : 


你 所 看 到 的 具体 输出 取决 于 PHP 错 误 设 置 。 也 就 是 说 ， 可 能 会 也 可 能 不 会 产生 如 图 7-3 所 
示 的 “notice”， 但 是 ， 它 会 示意 在 “test variable” 的 后 面 缺 少 一 个 附加 的 字符 串 。 


变量 $testvariable 的 值 没 有 显示 ， 因 为 在 testO 函 数 之 外 ， 这 个 变量 是 
MEE. AS 了， 在 第 5 行 试 图 访问 一 个 不 存在 的 变量 ， 只 有 PHP 设 
置 设 为 显示 所 有 的 错误 、 提 示 和 敬告 的 时 候 ， 才 会 产生 一 个 如 图 7-3 所 
示 的 提示 。 如 果 你 的 错误 设置 是 没有 恬 格 设置 ， 将 只 会 显示 字 从 串 “test 


variable:” « 
同样 地 ， 在 函数 外 声明 的 变量 将 不 能 目 动 在 函数 中 使 用 。 


使 用 global 语 句 访 问 变 量 


从 一 个 函数 内 部 ， 我 们 不 能 默认 地 访问 在 另 一 个 函数 中 或 在 脚本 中 
其 他 地 方 定 义 的 变量 。 在 一 个 函数 内 部 ， 如 朵 试图 使 用 具有 相同 名 子 的 


一 个 变量 ， 你 只 能 设置 或 访问 局 部 变量 。 让 我 们 在 程序 清单 7.6 中 验证 


程序 清单 7.6 ”默认 情况 下 ， 在 函数 之 外 定义 的 变量 不 能 在 函数 中 访问 


: <?php 

: $life = 42; 

function meaningOfLife() 
{ 


echo "The meaning of life is ".$life"; 


: meaningOfLife() ; 
?> 


OnNaOoah wah 一 


把 上 述 代码 放 入 到 一 个 名 为 scopetest2.php 的 文本 文件 中 ， 并 且 将 这 
个 文件 放置 在 Web 服 务 器 文档 根 目 录 下 。 当 通过 Web 浏 览 右 来 访问 这 个 
脚本 的 时 候 ， 它 产生 如 图 7-4 所 示 的 结果 。 
































/ fa] localhost/07/scopetest?. phy : 











| T © |O http://localhost, #/07/scopetest2.p php > | 


Notice: Undefined vanable: hfe in C:\xampp\htdocs\07\scopetest2.php on line 
4 
The meaning of life is 








图 7-4 ”试图 在 一 个 函数 的 作用 域外 引用 一 个 变量 


正如 你 所 预料 的 ，meaningOfLifeO 函 数 不 能 访问 在 第 2 行 定义 的 $life 
变量 ， 当 函数 试图 显示 $life 变 量 的 值 的 时 候 ， 它 是 空 的 。 忆 的 来 说 ， 这 
是 一 件 好 事 ， 因 为 它 避 免 了 我 们 在 同名 的 变量 之 则 的 潜在 的 冲突 ， 并 且 
如 果 一 个 函数 需要 外 界 的 信息 ， 它 可 以 通过 一 个 参数 。 偏 尔 ， 你 可 能 硕 
望 在 一 个 函数 内 部 访问 一 个 重要 的 变量 ， 而 又 不 想 将 它 作 为 一 个 参数 传 
圳 进来 ， 这 就 是 global 语 句 的 用 武之 地 。 程 序 清早 7.7 使 用 global 来 使 一 
切 恢 复 正 常 。 


程序 清单 7.7 使 用 global 语 句 访 问 全 局 变量 


: <?php 
: $life=42;3 
: function meaningOfLlife() 
{ 
global $life; 
echo "The meaning of life is ".$life"; 


} 
: meaningOfLife() ; 
?> 


OmonN Ooh WND 一 


把 上 述 代码 放 入 到 一 个 名 为 scopetest3. Ep 并 且 将 这 
个 文件 放置 在 Web 服 务 套 文档 根 目 录 下 。 当 通过 Web 浏 览 厚 来 访问 这 个 
脚本 的 时 候 ， 它 产生 如 图 7-5 所 示 的 结 





| [Š] localhost/07/scopetest3.ph 


€ © |© http://localhost/07/scopetest3.php x, | 


The meaning of life is 42 


图 7-5 ”在 一 个 函数 中 ， 使 用 global 语 句 成功 地 访问 一 个 全 局 变量 


当 我 们 在 meaningOfLifeO 函 数 中 声明 $life 变 量 的 时 候 〈 第 4 行 ) i 
过 把 global 语 句 放 置 在 它 前 面 ， 吏 可 以 引用 在 函数 的 外 面 〈 第 2 行 ) 声明 
的 $life 变 量 了 。 在 需要 访问 一 个 特别 指定 的 全 局 变量 的 函数 中 ， 都 需要 
使 用 global 语 句 。 但 是 请 留意 ， 如 条 在 函数 中 操作 变量 的 内 容 ， 变 量 的 
值 可 能 会 在 整个 脚本 的 范围 内 都 友 生 变化 。 


我 们 可 以 使 用 gobal 语 名 一 次 声明 多 个 变量 ， 只 需要 用 逗号 将 想 要 
访问 的 每 个 变量 隔 开 ， 示 例如 下 。 


global $var1, $var2, $vars3; 








通 第 ， 参 数 束 是 调用 代码 所 传递 的 任何 值 的 一 个 副本 ， 在 函数 中 修改 它 ， 对 于 函数 块 以 
外 的 部 分 没有 任何 影响 。 为 一 方面 ， 在 一 个 函数 中 修改 一 个 全 局 变量 ， 则 会 修改 原始 值 而 不 


是 副本 。 因 此 ， 使 用 global 语 句 的 时 候 要 小 心 。 





7.6 ”使 用 static 语 名 在 水 数 调 用 之 间 你 存 状 态 


疯 数 中 的 局 部 变量 拥有 一 个 短暂 而 快乐 的 人 生 ， 它 们 产生 于 函数 调 
用 的 时 候 ， 而 当 函 数 执 行 完成 的 时 候 它 们 束 疹 鼎 了 了。 


然而 ， 偶 尔 我 们 也 会 需要 给 函数 一 个 基本 的 和 内存。 假设 我 们 需要 一 
个 冰 数 来 记录 它 已 经 个 调 用 的 次 数 ， 以 便 一 个 脚本 可 以 创建 一 个 次 数 标 
题 。 我 们 当然 能 使 用 global 语 句 做 到 这 一 点 ， 如 程序 清单 7.8 所 示 。 


程序 清单 7.8 使 用 global 语 句 在 函数 调用 之 间 记 录 一 个 变量 的 值 


<?php 
$num of calls = ð; 
function numberedHeading($txt) 


global $num of calls; 

$num of callstt+; 

echo "<hi>".$num_of_calls." ".$txt."</h1>"; 
} 
9: numberedHeading ("Widgets"); 
10: echo "<p>We build a fine range of widgets.</p>"; 
11: numberedHeading( "Doodads"); 
12: echo “<p>Finest in the world.</p>"; 
13: ?> 


ON oath ah 一 


把 上 述 代码 放 入 到 一 个 名 为 numberedheading.php 的 文本 文件 中 ， 并 
且 将 这 个 文件 放 在 Web 服 务 需 文档 根 目 孙 下 。 当 通过 Web 浏 览 需 来 访问 
这 个 脚本 的 时 候 ， 它 将 产生 如 图 7-6 所 示 的 结果 。 





| fa localhost/07/numberedhez 


= © @ hittp://localhost/07/numberedheading.php A 
1 Widgets 
We build a fine range of widgets. 


2 Doodads 


' Finest m the world. 


Al7-6 ”使 用 global 语 句 来 记录 一 个 函数 调用 的 次 数 


这 确实 有 效 。 我 们 在 numberedHeadingO 函 数 的 外 部 ， 即 第 2 行 声明 
了 一 个 变量 $num_of_calls。 在 第 5 行使 用 一 条 global 语 句 使 这 个 变量 变 得 
对 函数 可 用 。 


每 次 调用 numberedHeading()，$num_of_calls 的 值 都 会 增加 1 在 第 6 
ÍT) ， 然 后 我 们 可 以 使 用 正确 的 、 递 增 后 的 标题 编号 来 显示 这 个 标题 。 


然而 ， 这 还 不 是 最 优雅 的 解决 方案 。 使 用 global 语 句 的 函数 不 能 作 
为 独立 的 代 人 码 段 阅读 。 在 阅读 或 复 用 这 些 函 数 的 时 低 ， 我 们 需要 小 心 它 
们 所 操作 的 全 局 变量 。 

这 种 情况 下 ，static 语 句 耽 派 上 用 场 了 了。 如 采 你 在 一 个 函数 中 使 用 
static 语 人 句 声 明了 一 个 变量 ， 这 个 变量 对 于 诠 图 数 仍 保持 为 局 部 的 ， 并 且 
图 数 在 从 一 次 执行 到 画 一 次 执行 的 过 程 中 会 * 记 住 ?该 变量 的 值 。 程 序 清 


单 7.9 修 改 了 程序 清单 7.8 的 代码 ， 并 且 使 用 了 static 语 人 句 。 
程序 清单 7.9 ”使 用 static 语 句 在 函数 调用 之 间 记 住 一 个 变量 的 值 


: <?php 
function numberedHeading($txt) 


static $num of calls = 0; 

$num_of_calls+t+; 

echo: shi” Saunt oT Calis. * ™ SEKT Sires 
} 
numberedHeading( "Widgets" ); 
9: echo "<p>We build a fine range of widgets.</p>"; 
1@: numberedHeading( "Doodads"); 
11: echo "“<p>Finest in the world.</p>"; 
12x T> 


OnNooahr wh — 


numberedHeading() Mi 2i CASS Ce AES. SERATA H 
$num_of_calls 变 量 时 ， 我 们 束 把 初始 值 赋 给 了 它 。 当 函数 在 第 8 行 第 一 
次 调用 的 时 候 ， 这 个 赋值 就 进行 了 。 当 函数 在 第 10 行 第 二 次 调用 的 时 
候 ， 这 个 最 初 的 赋值 被 忽略 了 ， 相 反 ， 人 代码 记 住 了 $num_of calls 的 前 一 
个 值 。 我 们 现在 可 以 把 numberedHeadingO 函 数 粘 贴 到 其 他 的 脚本 中 ， 而 
个 需要 担心 全 局 变量 。 尽 官 程序 清单 7.9 的 输出 确实 和 程序 清单 7.8 相 
辣 ， 但 我 们 使 代码 更 为 优雅 。 


7.7 关于 参数 的 更 多 内 容 


我 们 已 经 看 到 了 如 何 把 参数 传递 给 函数 ， 但 是 这 还 不 够 。 在 本 节 
中 ， 我 们 将 会 看 到 一 种 给 参数 设置 玲 认 值 的 扩 术 ， 并 且 找 到 一 种 通过 引 
用 而 不 是 通过 值 来 传递 参数 值 的 方法 。 传 递 引用 意味 看 ， 给 了 函数 参数 
一 个 最 初 值 的 别名 而 不 是 它 的 一 个 副本 。 


7.7.1 Wur E AME 


PHP 拥 供 了 一 个 极 好 的 功能 ee 数 。 到 目前 为 止 ， 我 
们 已 经 所 到 一 些 函 数 需 要 一 个 或 多 个 参数 ， 通 过 使 条 些 参数 变 为 可 选 
的 ， 我 们 可 以 让 函数 少 一 些 专 有 性 。 


程序 清单 7.10 创 建 了 一 个 有 有 用 的 小 函数 ， 它 把 一 个 字符 串 包谷 在 一 
个 HIML 的 Span 元 素 中 。 我 们 想 要 给 函数 和 用 户 一 个 能 够 修改 font-size 样 
式 的 机 会 ， 因 此 ， 除 了 字符 串 之 外 我 们 还 需要 一 个 参数 $fontsize (第 2 
ÍT) à 
程序 清单 7.10 ”需要 两 个 参数 的 一 个 函数 
<?php 
function fontWrap($txt, $fontsize) 
echo "<span style=\"font-size:$fontsize\">".$txt. "</span>"; 
} 
fontWrap("A Heading<br/>","24pt"); 
fontWrap("some body text<br/>","16pt"); 
fontWrap("smaller body text<br/>","12pt"); 


fontWrap("even smaller body text<br/>","10pt") ; 
0: ?> 


— OOnN OAR UON 一 


把 上 述 代 码 放 入 到 一 个 名 为 fontwrap.php 的 文本 文件 中 ， 并 且 将 这 


AS SCE TE WebdhS 28 CRSA A Se Ri Webwl bi 485K 7 eax 
脚本 的 时 候 ， 它 将 产生 如 图 7-7 所 示 的 结果 。 





SE EE x 
/ [A] localhost/07/fontwrap.php x 4 
| < C | ®© http://localhost/07/fontwrap.php as | 
A Heading 
some body text 
smaller body text 


even smaller body text 


图 7-7” 格 去 化 并 输出 字符 是 的 一 个 函数 


退 过 在 函数 定义 的 括 写 中 把 一 个 值 赋 给 一 个 参数 变量 ， 我 们 可 以 使 
$fontsize 参 数 变 为 可 选 的 。 如 果 函 数 调用 时 没有 为 这 个 参数 传递 一 个 参 
数值， 我 们 定义 时 赋 给 参数 的 值 吏 将 被 采用 。 程 序 请 单 7.11 使 用 这 种 技 
术 来 使 得 $fontsize 参 数 变 为 可 选 的 。 


程序 清单 7.11 市 有 一 个 可 选 参数 的 函数 


<?php 
function fontWrap($txt, $fontsize = "12pt") 
{ 


RON + 


echo "<span style=\"font-size:$fontsize\">".$txt. "</span>"; 


} 

fontWrap("A Heading<br/>","24pt"); 

fontWrap("some body text<br/>"}; 

fontWrap("smaller body text<br/>"); 

fontWrap("even smaller body text<br/>"}; 
n TS 


- Oa nN Oo oi 


© 


“4 FS Bl A fontWrap() Rž, AREA TR, 327 
参数 值 用 来 设置 span 元素 的 fontrsize 属 性 。 当 我 们 忽略 了 这 个 参数 ， 如 
第 7 行 、 第 8 行 和 第 9 行 所 示 ， 残 使 用 默认 值 *12pt"。 可 以 创建 任意 多 个 可 
选 参数 ， 但 是 当 我 们 给 一 个 可 选 参数 赋 一 个 默认 值 时 ， 所 有 后 续 的 参数 
tH ABV IZ 2 XE BRUM -o 


7.7.2 ”把 变量 引用 传递 给 函数 


当 我 们 把 参数 传递 给 函数 时 ， 它 们 作为 副本 存储 在 参数 变量 中 。 在 
图 数 体 中 对 这 些 变量 的 任何 修改 都 是 局 部 的 ， 并 用 不 会 反映 到 图 数 之 
外 ， 这 一 点 可 以 通过 程序 请 单 7.12 来 说 明 。 


程序 清单 7.12 ”通过 值 来 把 参数 传递 给 一 个 函数 


:<?php 
: function addFive($num) 


{ 
} 


$orignum = 19; 

: addFive($orignum) ; 
echo $orignum; 

?> 


$num += 5; 


OAOAnN DO HRWNDM — 


把 上 述 代码 放 入 到 一 个 名 为 addfive.php 的 文本 文件 中 ， 并 且 将 这 个 
文件 放置 在 Web 服务 器 文档 根 目 录 下 。 当 通过 Web 浏 览 器 来 访问 这 个 
脚本 的 时 候 ， 它 将 产生 如 下 结果 。 


19 


addFive() K 22252 — “7S A SESE A EES, (Bee Aik [al 
什么 。 在 第 6 行 我 们 把 一 个 值 赋 给 一 个 变量 $orignum， 然 后 在 第 7 行 把 这 
个 变量 传递 给 addFive() 函 数 ，$orignum 内 容 的 一 个 副本 存储 在 变量 $num 
中 ， 尽 管 我 们 把 $num 增 加 了 5， 但 这 对 于 $orignum 的 值 还 是 没有 影 啊 。 
当 我 们 显示 $orignum， 会 看 到 其 值 仍 然 是 10。 上 默认 情况 下 ， 传 递 给 函数 
的 变量 是 根据 值 来 传递 的 。 换 句 话 说 ， 制 造 了 变量 的 值 的 本 地 副本 。 


我 们 可 以 通过 创建 一 个 对 初始 变量 的 引用 来 改变 这 种 行为 ， 可 以 把 
一 个 引用 理解 成 指 癌变 量 的 一 个 路 标 。 在 使 用 引用 的 时 候 ， 我 们 操作 它 
所 指 问 的 变量 的 值 。 


程序 清单 7.13 展 示 了 这 一 扩 术 。 当 我 们 通过 引用 把 一 个 参数 传递 给 
一 个 函数 的 时 候 ， 如 第 7 行 所 示 ， 我 们 所 传递 的 变量 〈$orignum) 的 内 
容 在 图 数 中 被 这 个 参数 变量 访问 并 操作 ， 而 不 只 是 变量 的 值 的 一 个 副本 
(10) 家 访问 和 操作 。 在 这 些 情 况 下 ， 对 一 个 参数 的 任何 改变 都 会 改变 
最 初 变 量 的 值 。 我 们 可 以 通过 在 函数 定义 中 的 参数 名 的 前 面 洪 加 一 个 & 
符号 ， 从 而 用 引用 来 传递 一 个 变量 ， 如 第 2 行 所 示 。 


程序 清单 7.13 ”使 用 一 个 函数 定义 ， 把 参数 通过 引用 传递 给 函数 


: <?php 
: function addFive(&$num) 
{ 
$num += 5; 
} 
: $orignum = 10; 
: addFive($orignum); 
: echo $orignum; 
?> 


Oman gO 上 hm 一 


把 上 述 代 码 放 入 到 一 个 名 为 addfive2.php 的 文本 文件 中 ， 并 且 将 这 
个 文件 放置 在 Web 服 务 右 文档 根 目 录 下 。 当 通过 Web 浏 览 右 来 访问 这 个 


AAS EEE, CREU RAR. 
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7.8 MARA E a EE 


Se A Ved PRC A, RIIK A ee A 1 PB BE a EE» PHP | 
TETERA 忆 仿 不同 的 功能 ， 如 来 所 编 en ng 

上 运行 ， 我 们 可 能 需要 验证 关键 功能 是 合 可 用 。 例 如 ， 你 可 能 想 要 编 
ei 如 末 MySQL 相 关 的 功能 可 以 使 用 的 话 束 使 用 MySQL， 
侍 则 只 古人 简 蛙 地 把 数据 记录 到 一 个 文本 文件 中 ，。 


可 以 使 用 function_existsO 函 数 来 检查 函数 的 可 用 性 。 
需要 一 个 表示 图 数 名 的 字符 串 ， 如 果 可 以 找到 该 函数 ， 


function_exists() 


返回 true， 合 则 返回 false。 


修 


程序 清单 7.14 示 意 了 function_existsO 函 数 的 使 用 ， 并 且说 明了 我 们 
在 本 章 中 介绍 到 的 其 他 的 一 些 主题 。 


程序 清单 7.14 测试 一 个 函数 的 存在 性 


1: <?php 

2: function tagWrap($tag, $txt, $func = "") 

a: | 

4: if ((tempty($txt)) && (function exists($func))) { 
5: $txt = $func($txt); 

6: refurm “eT ea. is weeks Sw Leu. SSDP 3 
os } else { 

8: return "<strong>".$txt."</strong><br/>"; 

9 : } 

10: } 


12: function underline($txt) 


13: { 

14: return "<span style=\"text-decoration: underline; \">".$txt."</span>" ; 
15: } 

16: echo tagWrap('strong', ‘make me bold'); 

17: echo tagWrap('em', ‘underline and italicize me', “underline"} ; 

18: echo tag“Wrap('em', ‘make me italic and quote me’, 

19: create function('$txt', ‘return "“&quot;$txt&quot;";')); 

20: ?> 


我 们 定义 了 两 个 函数 ，tagWrapO 《在 第 2 行 ) 和 underline()〈 在 第 12 
ÍT) o tagWrap( KNEZI TS AA: 一 个 标记 、 要 格式 化 的 文本 以 及 一 
个 可 选 的 六 数 名 ， 它 返回 一 个 格式 化 后 的 字符 串 。underline(0) 函 数 需 要 
一 个 参数 ， 即 要 格式 化 的 文本 ， 并 且 返 回 包 含 在 <span> 标 记 中 的 文本 ， 
而 <sSpan> 标 记 珊 有 相应 的 样式 属性 。 


当 在 第 16 行 第 一 次 调用 tagWrapO 的 时 候 ， 我 们 把 字符 'strong' 和 字 
侍 串 “make me bold” 传 递 给 它 。 由 于 我 们 没有 给 函数 参数 传 递 一 个 值 ， 
就 使 用 默认 的 值 〈 一 个 空 字 符 串 ) 。 在 第 4 行 ， 我 们 检查 $txt 变 量 是 否 

含 字符 串 ， 以 及 $func 是 耕 存 在 ， 我 们 调用 function_exists() 来 根据 这 个 
名 字 检 栓 一 个 函数 。 当 然 ， 在 这 个 例子 中 ，$func 变 量 是 空 的 ， 因 此 ， 
我 们 在 第 7 到 8 行 的 else 子 句 中 把 $txt 变 量 包含 到 <strong> 标 记 中 并 返回 结 
果 。 


我 们 在 第 17 行 使 用 字符 串 ‘“em?*、 某 些 文本 以 及 第 三 个 参 


#“underline” Val FtagWrap(). function_exists() f+ 44 Aunderline() H PK 2 
(在 第 12 行 ) ， 于 是 它 调 用 这 个 函数 ， 并 且 在 进行 任何 进一步 的 格式 化 
之 朋 把 参数 变量 $txt 传 递 给 它 。 结 果 是 一 个 斜体、 市 下 划 线 的 字符 串 。 


最 后 ， 在 第 18 行 ， 我 们 调用 tagWwrapO0， 它 把 文本 放 到 引用 实体 中 。 
直接 给 要 改变 的 文本 添加 实体 ， 这 应 该 更 快 ， 但 是 ， 这 个 示例 主要 用 来 
说 明 一 点 ， 即 function_existsO 国 数 对 于 匿名 函数 和 对 于 表示 函数 名 的 字 
从 串 一 样 有 效 。 

把 上 述 代 码 放 入 到 一 个 名 为 exists.php 的 文本 文件 中 ， 并 且 将 这 个 文 
件 放 置 在 Web 服 务 占 文档 根 目录 下 。 当 通过 Web 浏 贤 右 来 访问 这 个 脚本 
的 时 候 ， 它 将 产生 如 图 7-8 所 示 的 结果 。 





[a] localhost/O///e0sts.php 
i CG | © htto://localhost/07/exists.ohp a 
make me bold 


underline and italicize me 
"make me italic and quote me" 


图 7-8 ”exists.php 的 输出 
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KENA SARRAR, URUTE EN. RETZ F an 
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数 的 存在 性 。 


7.10 Q&A 


Q: HUESA SRE Sg eT BP Le PS eR Bd H 
H, WARI 7 ee IT ABLE AB PE 2 


A: BALL. BANDAS] Sob A eRe. PATO, RATE AE 
FBT, FP AE ER Be ACE ZB), EE PRT EAE 17 
连接 起 来 ， 示 例如 下 。 


$newstring = "I purchased" .numPurchase({$somenum)." items."; 


Q: WRB AP AE RR, KERTH PS A 
便 用 的 名 字 来 声明 一 个 函数 ， 会 妥 生 什么 情况 ? 


A: 调用 一 个 不 存在 的 函数 或 者 使 用 和 其 他 已 有 的 函数 同样 的 名 字 
来 声明 一 个 函数 ， 将 会 导致 脚本 停止 执行 。 浏 览 需 中 是 个 显示 一 条 错误 
消息 ， 取 雇 于 你 的 php.ini 文 件 中 的 错误 设置 。 


7.11 实践 练习 


这 个 实践 练习 设计 用 来 帮助 你 预测 可 能 的 问题 、 复 习 已 经 学 过 的 知 
识 ， 并 且 开 始 把 知识 用 于 实践 。 


1. 判断 对 错 : 如 果 一 个 函数 不 需要 一 个 参数 ， 你 可 以 在 函数 调用 
HEER S o 


2. 如 何 从 函数 返回 一 个 值 ? 


3. 如 下 代码 段 将 回 浏 宽 冀 显示 什么 ? 


$number = 50; 


Function tenTimes({) { 
$number = $number * 18; 


} 


tenTimes(}; 
echo $number; 


4. On RAMS BCR IA ba NIT ? 
$number = 58; 
Function tenTimes({) { 

global $number; 


$number = $number * 18; 


} 


tenTimes(}; 
echo $number; 


5. UN RAMS BCH Ri iia ANITA ? 


$number = 50; 


Function tenTimes( &$n ) { 
$n = $n * 14; 
} 


tenTimes{ $number }; 
echo $number; 


oy 


ee 
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2. 必须 使 用 return 天 键 字 。 


3. 它 显 未 50。tenTimes() 函 数 没 有 访问 全 局 变量 $number。 当 调用 
该 函数 的 时 候 ， 它 将 会 操作 目 己 的 局 部 变量 $number。 


4. 它 显 示 500。 我 们 使 用 了 了 global 语句 ， 这 会 让 tenTimesO 函 数 访 问 
外 部 $number 变 量 。 


5. 它 显示 500。 通 过 加 参数 变量 gn 添 加 人 符号 ， 我 们 确保 了 这 个 参 
inate, $n 和 $number 指 同 同 一 个 值 ， 因 些 ， 对 $n 的 任何 改变 都 
将 在 $number 反 映 出 来 。 


BA Wel 


1， 回 顾 创建 一 个 函数 的 语法 ， 该 函数 接受 参数 、 操 作 这 些 值 ， 并 
用 返回 一 个 字符 串 。 


2. 创建 一 个 水 数 ， 它 接受 4 个 字符 串 变 量 并 且 返 回 一 个 字符 串 ， 其 
中 包含 了 一 个 HTML 表 格 元 系 ， 把 每 个 变量 都 放置 在 自己 的 单元 格 中 。 


Heit ”使 用 数组 


ERE, RITKA Od: 


。 如 何 创 建 天 联 数 组 和 多 维 数 组 。 
。 如 何 使 用 PHP 内 建 的 众多 和 数组 相关 的 函数 。 


数组 用 来 存储 和 组 织 数 据 。PHP 包 括 很 多 与 数组 相关 的 函数 ， 这 些 
孙 数 使 得 你 可 以 创建 、 修 改 和 操作 数组 ， 而 数组 将 在 本 书 所 拍 述 的 过 程 
式 程序 设 计 方 法 中 频 喧 地 使 用 。 


8.1 什么 是 数组 


在 本 书 前 面 的 各 草 ， 我 们 已 经 学 习 并 使 用 了 标量 变量 ， 并 且 知 道 这 
些 变 量 用 来 存储 什 。 但 是 ， 标 量变 量 一 次 只 能 存储 一 个 什 ， 如 $color 变 
量 只 能 存储 一 个 red 值 或 blue 值 等 ， 但 它 无 活用 来 存储 彩虹 中 的 两 色 列 
表 。 而 数组 是 一 种 特殊 类 型 的 变量 ， 它 使 得 我 们 能 够 存储 任意 多 个 值 ， 
包括 彩虹 中 的 所 有 7 种 闫 色 。 


尽管 可 以 在 一 个 数组 中 存储 尽 可 能 多 的 值 ， 但 一 些 数 组 函数 还 是 有 
100000 个 值 的 上 限 。 如 果 在 数组 中 要 存储 大 量 的 数据 ， 确 保 阅 读 PHP 手 
册 中 有 关 你 要 使 用 的 数组 函数 的 条 目 ， 看 看 该 函数 是 否 有 操作 数据 的 上 
限 。 


数组 是 有 和 系 引 的 ， 这 意味 看 每 个 条 目 部 由 一 个 键 key) 和 一 个 值 
(value) 组成。 键 是 索引 的 位 置 ， 从 0 开始 并 且 对 于 数组 中 的 每 个 新 元 
系 者 增加 1。 值 惑 是 我 们 和 该 位 置 关 联 起 来 的 任何 值 : PB — 
个 整数 或 任何 我 们 希望 的 值 。 可 以 把 一 个 数组 看 作 是 一 个 文件 柜 ， 而 一 
个 键 / 值 对 束 是 一 个 文件 夹 ， 键 就 是 文件 夹 上 面 的 标签 ， 而 值 束 古文 件 
来 中 的 文件 。 当 我 们 在 下 一 节 中 创建 数组 的 时 候 ， 残 会 看 到 这 种 类 型 的 
结构 的 实际 应 用 。 


8.2 ”创建 数组 


我 们 可 以 使 用 array0 函 数 或 者 数组 操作 符 [来 创建 一 个 数组 。 当 我 
们 想 要 一 次 性 创建 一 个 新 的 数组 并 且 用 多 个 元 素来 填充 它 的 时 候 ， 通 毅 
使 用 array0 函 数 。 当 我 们 想 要 创建 一 个 狐 的 数组 并 且 它 开始 的 时 候 只 有 
一 个 元 妹 ， 或 者 当 我 们 想 要 添加 一 个 已 经 存在 的 数组 元 妹 的 时 候 ， 通 各 
使 用 数组 操作 符 


如 下 的 代码 段 显示 了 如 何 使 用 array0 函 数 创 建 一 个 名 为 $rainbow 的 
AA, KFS UMARE. 


$rainbow = array{"red", "orange", "yellow", "green", "blue", "indigo", 
"yliolet"}; 


如 下 的 代码 段 展示 了 使 用 数组 操作 符 来 逐渐 地 创建 同一 个 数组 的 方 


$rainbow[] = "red"; 
$rainbow[] = "orange"; 
$rainbow[] = "yellow"; 
$rainbow[] = "green"; 
$rainbow[] = "blue"; 
$rainbow[] = "indigo"; 
$rainbow[] = "violet"; 


这 两 个 代码 段 都 创建 了 一 个 名 为 $rainbow 的 7 元 系 的 数组 ， 其 值 从 
索引 位 置 0 开 始 到 索引 位 置 6 结 束 。 如 末 你 想 要 排列 出 它们 ， 可 以 指定 索 
引 位 首 ， 像 下面 这 样 编 写 代 但 。 


$rainbow[@] = "red"; 
$rainbow[1] = "orange"; 
$rainbow[ 2] = "yellow"; 


G$rainbow[3] = "green"; 
Srainbow[4] = "blue"; 
G$rainbow[5] = "indigo"; 
Srainbow[6] = "violet"; 


然而 ， 没 有 指定 位 置 的 时 候 ，PHP 会 为 你 做 这 些 ， 并 且 会 消除 像 下 
面 的 例子 这 样 排 错 元 系 的 可 能 性 。 


$rainbow[0] = "red"; 

$rainbow[1] = "orange"; 
$rainbow[2] = "yellow"; 
$rainbow[5] = "green"; 
$rainbow[6] = "blue"; 

$rainbow[7] = "indigo"; 
$rainbow[8] = "violet"; 


AE RY ENEH array) A BOR xe BA BRE FT BSE BUA, Ue DA Ase 
用 效 组 操作 符 来 添加 数组 。 在 下 面 的 第 一 行 中 ，6 个 元 系 讨 加 到 了 数组 
中 ， 在 第 二 行 ， 万 外 一 个 元 系 添 加 到 了 数组 的 末尾 。 


$rainbow = array("red', "orange", "yellow", "green", "blue", "indigo"); 
S$rainbow[] = "violet"; 


AS“ PEHA ee 8 A, BERNE LEN BZ SSA 
在 下 面 两 万 中 ， 我 们 将 学 习 两 种 其 他 闫 型 的 数组 ; 关联 数组 和 多 维 数 
组 。 


8.2.1 创建 关联 数组 


数字 索引 数组 使 用 一 个 索引 位 置 作为 键 ， 如 0、1、2 等 ， 而 关联 数 
组 使 用 实际 命名 的 键 。 下 面 的 例子 通过 创建 一 个 具有 4 个 元 聚 的 名 为 
$character 的 数组 来 说 明 这 一 点 。 


$character = array{ 
"name" => "Bob" i 
"occupation" => "superhero", 
age => 30, 
"special power" => "x-ray vision" 
); 
$character 数 组 中 的 4 个 键 是 name、occupation、age 和 special power. 
天 联 的 值 分 别 是 Bob、superhero、30 和 x-ray vision。 我 们 可 以 使 用 指定 


的 键 来 引用 关联 数组 的 具体 元 系 ， 如 下 面 的 例子 所 示 。 
echo $character|[ ‘occupation’ ] í 
上 述 代码 段 的 输出 如 下 。 
superhero 
ANA FR S| ZA iE, RITE LME B28 BRE RRI RRL 
组 ， 示 例如 下 。 
S$character['supername'] = "Mega X-Ray Guy"; 


这 个 例子 添加 了 一 个 名 为 Supername 的 键 ， 其 值 为 Mega X-Ray 
Guy 。 


一 个 天 联 数 组 和 一 个 数字 索引 数组 之 间 的 唯一 的 区 别 束 是 键 名 ， 在 
一 个 数字 索引 数组 中 ， 键 名 是 数字 。 在 一 个 关联 数组 中 ， 键 名 是 一 个 有 
意义 的 单词 。 
8.2.2 ”创建 多 维 数 组 


有 前面 介绍 的 两 种 数组 分 别 存储 字符 串 和 整数 ， 而 这 里 介绍 的 数组 则 
存储 其 他 的 数组 。 如 果 每 组 键 / 值 对 构成 了 一 维 ， 一 个 多 维 数组 存储 了 
多 组 这 样 的 键 / 值 对 。 人 例如， 程序 清 单 8.1 定 义 了 一 个 名 为 $characters 的 多 
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但 是 ， 它 实际 上 只 是 包含 其 他 数组 的 一 个 数组 。 


程序 清单 8.1 定义 一 个 多 维 数组 


1 <?php 

2 $characters = array{ 

3 array ( 

4: "name" => "Bob", 

ae "occupation" => "superhero", 

6: "age" => 30, 

7 "special power" => "x-ray vision" 
8 E 

9: array ( 

10: "name" => "Sally", 

i E- "occupation" => "superhero", 

Tas "age" => 24, 

t3: "special power" => "superhuman strength" 
14: Vs 

15: array ( 

16: "name" => "Jane", 

17: "occupation" => "arch villain", 
18: "age" => 45, 

19: "special power" => "nanotechnology" 
20: ) 

上 ); 

22: T> 


fe 247, AE Haray K 29) 4a atzA $characters. 333~847 AN J 
$characters 数 组 的 第 一 个 元 素 ， 第 9 一 14 行 显示 了 $characters 数 组 的 第 二 
个 元 素 ， 而 第 15 一 20 行 显示 了 $characters 数 组 的 第 三 个 元 素 。 这 些 元 素 
可 以 用 $characters[0]、$characters[1] 和 $characters[2] 来 引用 。 


每 个 元 系 部 由 一 个 关联 数组 组 成 ， 这 个 关联 数组 日 映 包 含 4 个 元 


Za: name、occupation、age 和 special power. 
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echo $characters[1]; 


输出 的 结果 将 会 如 下 。 


Array 


因为 主 元 素 实 际 上 保存 了 一 个 数组 作为 其 内 容 。 要 想 真 正 地 得 到 想 
要 的 元 素 ， 即 在 内 部 数组 元 了 素 中 包含 的 具体 信息 ， 我 们 需要 访问 主 元 于 
索引 位 置 以 及 想 要 浏览 的 值 的 关联 名 。 


来 看 一 个 例子 。 


echo $characters[1][ Sccupation']; 


它 将 会 显示 如 下 结果 。 


superhero 


如 琳 把 如 下 的 代码 添加 到 程序 清单 8.1 的 末尾 ， 它 将 显示 出 存储 在 
每 个 元 系 中 的 信息 ， 并 在 浏 虹 右 中 多 显示 出 一 行 分 阳线 。 


foreach ($characters as $c} 1 
while {list($k, $v} = each ($c)) { 
echo "$k ... $v <br/>"; 
} 


echo "<hr/>"; 


foreach 循 环 和 主 数组 元 素 $characters 相 关 ， 它 过 历 这 个 数组 ， 并 且 
把 临时 变量 名 $c 赋 给 包含 在 每 个 位 置 中 的 元 素 。 接 下 来 ， 我 们 开始 一 个 
while 循 环 ， 这 个 循环 使 用 两 个 函数 来 提取 内 部 数组 的 内 容 。 首 先 ，listO 

浮 数 命名 了 占 位 从 变量 $k 和 $v， 这 两 个 变量 将 使 用 each() 疯 数 所 收集 的 
刍 和 值 来 填充 。each() 函 数 伍 看 $c 数组 的 每 个 元 素 并 且 相 应 地 提取 信 


= 


JNO 


echoi# fj H Æ wn EE H eachO AZt M $c 2 Zi FE AY AY BE BE AA 


($k#l$v) , JFAYSII—Aar hazel (Paz. B8- SK RAN 
mdarray.php 的 文件 的 结果 。 





| [E] localhost/08/mdarray.php  \\ 4 


| E c | @ http://localhost/08/mdarray.php A, 

























































































.. Bob 
occupation - . superhero 


es power ... X-Tay vision 





age ...45 
special power ... nanotechnology 


| 
| nam 
= 
fe 
| 
| 
name ... Sally 
| 


8.3 — HEA BZA AAR HI PK Z 


PHP 内 建 了 70 多 个 和 数组 相关 的 函数 ， 我 们 可 以 从 


http://www.php.net/array 了 解 详细 信息 。 本 市 介绍 其 中 一 些 很 党 用 的 ， 
并 且 很 有 用 的 函数 。 


è mon sizeof() 
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定 如 下 的 数组 

$colors = array("blue", "black", "red", "“green"}; 

count($colors); 和 和 sizeof($colors); 都 返回 值 4。 

ee 在 表 历 一 个 数组 并 返回 其 键 和 值 的 应 用 中 ， 这 两 
pe AOE Fy ECHL Uist0 古 看 上 去 像 一 个 函数 一 样 的 语言 结 

。 我 们 在 前 面 见 到 过 其 应 用 的 一 个 例子 ， 在 那里 ， 我 们 授 历 了 

$c 数组 并 显示 其 内 容 。 

foreach() 一 一 这 个 控制 结构 也 用 来 授 历 一 个 数组 ， 把 一 个 元 系 的 值 

赋 给 一 个 给 in 正如 我 们 在 上 一 和 中 见 到 的 。 

图 数 把 指针 返回 到 一 个 数组 的 开始 ， 示 例如 下 。 








reset() 


reset (S$character}; 


当 我 们 要 对 一 个 数组 执行 多 个 操作 的 时 候 ， 例 如 排序 、 提 取 值 等 ， 


这 个 函数 很 有 用 。 


array_push() 一 一 这 个 函数 在 一 个 已 有 数组 的 末尾 添加 一 个 或 多 个 元 
素 ， 示 例如 下 。 


array push($existingArray, “element 1", “element 2", "element 3"}; 


array_pop(0 一 一 这 个 函数 删 际 并 返回 一 个 已 有 数组 的 最 后 一 个 元 
A, ABI FP. 


$last_ element = array_ pop ($existingArray}; 


array_unshiftQ—_1% T RAE -NCA BZA FP RST PS 
Tu, AIO e 

array _unshift($existingArray, "element BE "element 2 ‘element es 
array_shift(0 一 一 这 个 函数 删除 并 返回 一 个 已 有 数组 的 第 一 个 元 率 ， 


例如 ， ao HS oS — PE A 7G 8 AY E 
$first _ element， 示 例如 下 。 


$first element = array snift(S$existingArray) ; 


函数 组 合 两 个 或 多 个 已 有 的 数组 ， 示 例如 


array_merge() 
Fe 
$newArray = array merge($array1, $array2); 

AZORE — AH, Pee SAA E AH 
acne, OT, 

$keysArray = array_keys{$existingArray}; 

图 数 返 回 一 个 数组 ， 其 中 包含 了 一 个 给 定数 
组 中 的 所 有 的 值 ， anes, 

$valuesArray = array values($existingArray); 

shuffle() KENE rE BZA A Tg JO a SLA. RS 
PR 


shuffle($existingArray} ; 








array_keys() 


array_values() 








XN BZA ASR HY BRC id ALB BOT TAE FOR JA ee a tE 


o PRIN, ZUH VA Re BZA AA RY R TE BE ASS USN i “Pb o> 


到 ， 因 此 ， 你 将 会 很 快 熟悉 它们 。 如 果 你 做 不 到 ， 位 于 
http://www.php.net/array 的 PHP 手 册 也 有 部 分 数组 可 供 参 考 ， 它 评 细 讨论 
了 和 数组 相关 的 所 有 函数 ， 包 括 排序 数组 的 十 多 种 不 同 的 方法 。 


8.4 Zi 


本 章 介 绍 了 数组 的 概念 ， 包 括 如 何 创建 和 引用 它们 。3 种 数组 类 型 
是 数字 索引 数组 、 关 联 数 组 和 多 维 数组 。 此 外 ， 我 们 看 到 了 PHP 内 建 的 
众多 数组 相关 函数 中 的 一 些 例子 。 这 些 函 数 可 以 用 来 操作 和 修改 已 有 的 
数组 ， 有 时 候 甚至 可 以 创建 一 个 全 新 的 数组 。 


8.9 Q&A 


Q: 多 维 数组 可 以 有 多 少 维 ? 


A: 可 以 在 多 维 数组 中 创建 任意 多 个 维 ， 但 是 列 志 了， 维 数 越 多 ， 
官 理 越 难 。 如 来 你 拥有 多 维 的 数据 ， 明 吞 的 方法 是 看 看 这 些 数 据 是 人 耕 可 
以 用 不 同 的 方法 存储 以 及 是 否 可 以 用 这 种 方法 访问 ， 例 如 在 数据 库 中 。 


Q: 如 末 只 是 想 要 创建 一 个 联系 表单 ， 为 什么 要 关心 数组 ? 


A: 即使 在 最 基本 的 客户 器 /服务 右 交 互 中 例如 ，Web 站 反 中 的 一 
MRA) ， 数 组 也 是 有 用 的 。 我 们 将 会 在 第 11 章 学 习 关 于 表 早 的 更 
多 知识 ， 但 是， 在 开始 使 用 本 章 中 的 知识 之 前 ， 有 儿 点 需要 记 住 。 如 末 
表单 包 台 了 任何 复 选 框 或 列表 ， 用 户 可 以 从 其 中 选择 多 个 选项 ， 那 么 ， 
这 些 数 据 应 该 作为 数组 发 送 给 表单 。 如 果 要 操作 这 些 数据 的 话 ， 你 需要 
将 数据 从 数组 中 取出 来 ， 本 章 给 出 了 几 个 基本 的 示例 惑 是 这 么 做 的 。 


8.6 ”实践 练习 


这 个 实践 练习 设计 用 来 帮助 你 预测 可 能 的 问题 、 复 习 已 经 学 过 的 知 
识 ， 并 且 开 始 把 知识 用 于 实践 。 


问答 题 


1. 什么 样 的 结构 可 以 用 来 定义 一 个 数组 ? 


2. 什么 函数 可 以 用 来 连接 两 个 数组 ? 


1. array() 


2. atray_merge() 


DA He 
L 回顾 定义 一 个 多 维 数组 并 引用 其 数据 项 的 过 程 。 


2. 创建 一 个 按照 类 型 组 织 的 电影 的 多 维 数组 。 它 将 会 采用 一 个 关 
联 数 组 的 形式 ， 电 影 的 类 型 如 Science Fiction、Action、Adventure 等 作为 
键 。 这 些 数 组 元 系 的 每 一 个 都 应 该 是 包含 电影 名 《例如 Alien、 
Terminator 3、Star Wars 等 ) 的 一 个 数组 。 创 建 了 数组 之 后 ， 遇 历 息 
们 ， 显 示 出 每 种 关 型 及 其 相关 电影 的 名 字 。 


BIE ”使 用 对 象 


ERER, MÄ FI: 


。 理解 一 个 对 象 的 基本 结构 。 
。 如 何 创建 和 操作 对 象 以 及 它们 所 包含 的 数据 。 


对 象 用 来 存储 和 组 织 数据 。 面 癌 对 象 编 程 是 这 样 一 种 编程 范 型 ， 其 
中 程序 (或 应 用 程序 〉 的 结果 围绕 看 这些 对 象 以 及 对 象 之 则 的 关系 和 交 
互 来 进行 。 很 多 程序 说 计 语 言 中 都 拥有 的 面 问 对 象 程 序 设计 结构 ， 而 这 
在 PHP 中 也 可 以 找到 。 


实际 上 ， 很 多 PHP 程 序 员 ， 尤 其 是 那些 具有 较 强 的 面向 对 象 编程 语 
言 背景 的 程序 员 ， 会 选择 以 面向 对 象 的 方式 来 开发 PHP 应 用 程序 ， 


然而 ， 在 PHP 中 ， 你 不 一 定 必须 用 和 面 癌 对 象 的 方式 来 编写 脚本 。 人 很 
多 PHP 脚 本， 实际 上 我 们 在 本 书 中 见 到 的 大 多 数 脚 本 ， 虱 是 过 程式 的 而 
不 是 面 问 对 象 的 。 也 束 是 说 ， 强 调 在 创建 程序 的 过 程 中 ， 使 用 变量 、 数 
据 和 控制 结构 、 例 程 和 函数 。 原 因 很 人 蚀 单 ， 如 末 你 全 新 地 接触 一 种 编 
程 ， 奢 么 ， 在 接触 一 种 本 里 有 很 多 相关 的 图 书 的 编程 范 型 之 前 ， 获 取 过 
程 芭 编 程 的 经 验 以 及 这 种 语言 的 基础 知识 是 很 重要 的 。 本 草 市 你 简单 地 
领略 一 下 对 象 的 世界 ， 因 为 对 于 你 继续 获取 更 多 的 知识 和 经 验 来 说 ， 这 
征 一 种 重要 的 数据 类 型 和 概念 。 


如 由 你 市 看 在 面 同 对 象 程序 设计 中 的 背景 来 学 习 PHP， 本 章 将 帮助 


你 理解 PHP 中 的 对 象 模型 。 


9.1 创建 一 个 对 象 


解释 一 个 对 象 的 概念 有 点 难 : 它 古 事物 的 一 个 理论 化 的 合子， 这些 
BY) EE, KAT) 都 存在 于 叫做 类 Class) 的 一 个 模板 式 结构 中 。 
尽管 很 容易 形象 地 描述 一 个 标量 变量 ， 例 如 带 有 red 值 的 $gcolor， 或 者 其 
中 带 有 3 个 或 4 个 不 同 元 素 的 名 为 gcharacter 的 一 个 数组 ， 某 些 人 还 是 在 形 
象 地 摘 述 对 象 的 时 候 感到 为 难 。 

从 现在 开始 ， 等 试 把 对 象 看 作 一 个 小 盒 季 ， 其 两 病 市 有 和 输入 和 和 输 
出 。 输 入 的 机 制 叫做 方法 (method) ， 方 法 具有 属性 (property) 。 在 
整个 本 章 中 ， 我 们 将 看 到 类 、 方 法 和 属性 是 如 何 协同 工作 产生 各 种 输出 
的 。 


提示 : 





如 果 你 对 类 的 概念 是 完全 陌生 的 ， 可 以 通过 阅读 PHP 手 册 的 “Classes and Object” ~ER 
实 自己 的 知识 。 可 以 在 http://www.php.net/manual/en/ language.oop5.php 找到 它 。 


本 节 开 始 的 时 候 提 人 到， 一 个 对 象 拥 有 一 个 叫做 类 的 结构 。 在 每 个 类 
中 ， 我 们 定义 了 一 组 特征 。 例 如 ， 假 设 你 已 经 创建 了 一 个 automobile 
类 ， 在 automobile 类 中 ， 我 们 应 该 有 color、make、model 特 征 。 每 个 
automobile 对 和 象 使 用 所 有 的 这 些 特征 ， 但 它们 都 初始 化 为 不 同 的 值 ， 例 
如 Silver、]Mazda 和 Protege5， 或 者 red、Porsche 和 Boxter。 


使 用 对 象 的 全 部 目标 就 是 创建 可 以 重用 的 代码 。 因 为 尖 古 如 此 案 密 
的 结构 但 义 羡 目 包 含 的 ， 并 且 彼 此 独立 ， 它 们 可 以 从 一 个 应 用 程序 重用 


到 另 一 个 应 用 程序 中 。 例 如 ， 假 设 你 编写 了 文本 格式 化 类 用 于 一 个 项 
目 ， 并 且 决 定 在 另 一 个 项 目 中 使 用 这 个 类 。 由 于 这 个 类 只 是 一 组 特征 ， 
你 可 以 取出 代码 并 且 在 第 二 个 项 目 中 使 用 它 ， 使 用 第 二 个 项 目 特定 的 方 
法 来 进入 其 中 ， 但 却 使 用 已 有 代码 的 内 部 工作 机 制 来 得 到 新 的 结果 。 


创建 一 个 类 很 简 早 ， 只 用 声明 其 存在 即 可 ， 示 例如 下 。 


class myClass { 
/fcode will go here 


I 


既然 已 经 有 了 一 个 类 ， 我 们 可 以 创建 对 象 的 一 个 新 实例 ， 示 例如 
Be 


$object] = new myClass(}; 


在 程序 清单 9.1 中 ， 你 已 经 证 实 了 对 象 的 存在 ， 即 便 其 中 没有 什么 
内 容 一 它 现 在 只 有 一 个 名 字 。 


程序 清单 9.1 证 实 对 象 的 存在 


<?php 
class myClass { 
/{/code will go here 
} 
$object1 = new myClass(); 
echo "\$objecti is an ".gettype($object1).".<br/>"; 


if (is object($objecti)) { 
echo "Really! I swear \$object1 is an object!"; 
: } 


ES 


— =<" OON OAAR ON — 
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如 果 将 上 述 代 码 保 存 为 proofofclass.php， 将 其 放置 在 文档 根 目录 
下 ， 使 用 Web 浏 贤 嚣 访问 它 ， 将 会 看 到 如 下 的 显示 结果 。 


$objecti is an object. 
Really! I swear $object1 is an object! 


这 不 是 一 个 特别 有 用 的 类 ， 因 为 它 并 没有 做 什么 事情 ， 但 它 是 有 效 
的 ， 并 且 通 过 第 2 行 到 第 5 行 展 示 了 闫 模板 是 如 何 工 作 的 。 第 8 行 到 第 10 
行使 用 is_object() 函 数 来 测试 菜 物 是 否 古 一 个 对 象 ， 在 这 个 例子 中 “和 
物 ” 束 是 $objectl1。 由 于 is_object() 的 测试 结果 为 tue，f 语 句 中 的 字符 串 
KER Bl] BE AEE o 


接 下 来 ， 我 们 将 学 习 对 象 的 属性 和 方法 ， 它 们 增加 了 类 模板 的 用 


i 


9.1.1 ”对象 的 属性 


在 对 象 中 声明 的 变量 叫做 属性 (property) 。 在 一 个 类 的 项 部 声明 
变量 ， 这 是 标准 的 做 法 。 这 些 属性 可 以 是 值 、 数 组 甚至 是 其 他 对 象 。 如 
下 的 代码 段 在 类 中 使 用 简单 的 标量 变量 ， 其 前 面 有 一 个 公共 关键 字 。 


class myCar { 
public$color = "silver"; 
public$make = "Mazda"; 
public$model = "Protege5"; 


如 果 你 在 变量 名 称 前 使 用 关键 字 public、protected 或 private， 那 么 ， 
可 以 表明 类 成 员 〈 变 量 ) 是 在 任意 地 方 都 可 以 访问 (public〉、 在 类 上 自 
身 或 父 类 或 继承 类 中 可 以 访问 (protected) 、 还 是 只 能 由 该 类 自身 访问 


(private) 。 


现在 ， 当 你 创建 一 个 myCar 对 象 ， 它 将 总 是 有 这 3 个 属性 。 程 序 清 
单 9.2 展 示 了 在 声明 这 些 属 性 并 且 给 它们 赋值 之 后 如 何 访 问 它们 。 


程序 清单 9.2 显示 对 象 属性 


1 <?php 

2: class myCar { 

a public$color = "silver"; 

4 public$make = "Mazda"; 

z public$model = "Protege5"; 

6: } 

7: $car = new myCar(); 

8: echo "I drive a: ".$car -> color." ".$car -> make." ".$car -> model; 
D: T> 


如 果 把 这 段 代 码 保 存 为 objproperties.php， 将 其 放置 在 文档 根 目 录 
下 ， 使 用 Web 浏 览 器 访问 它 ， 将 会 看 到 如 下 的 显示 结 


I drive a: silver Mazda Proteges 


由 于 你 总 是 开 着 一 辆 银灰 色 的 Mazda Protege5 的 机 会 很 小 ， 所 以 你 
想 要 改变 myCar 对 象 的 属性 。 程 序 清 单 9.3 示 意 了 如 何 做 到 这 一 点 。 


程序 清单 9.3 ”改变 对 象 属性 


1: <?php 

2: class myCar { 

3 public$color = "silver"; 
4: public$make = "Mazda"; 

5: public$model = "Protege5"; 
6: } 

7: $car = new myCar(); 

8: $car -> color = "red"; 

9: $car -> make = "Porsche"; 

10: $car -> model = "Boxter"; 

11: echo "I drive a: ".$car -> color." ".$car -> make." ".$car -> model; 
en i 


如 果 把 这 有 段 代码 保存 为 objproperties2.php， 将 其 放置 在 文档 根 日 录 
下 ， 使 用 Web 浏 览 器 访问 它 ， 将 会 看 到 如 下 的 显示 结 


I drive a: red Porsche Boxter 


在 这 个 例子 中 ， 即 便 $color、$make 和 $model 属 性 在 声明 的 时 候 没 有 初始 值 ， 第 8 行 到 第 10 
行 偿 是 会 为 它们 赋 一 个 值 。 只 要 属性 声明 了 ， 随 后 束 可 以 使 用 它们 ， 不 管 有 没有 初始 值 。 


程序 清单 9.3 的 目的 是 为 了 展示 只 要 拥有 了 定义 好 的 具有 属性 的 
类 ， 束 可 以 很 容易 地 修改 这 些 属性 的 值 以 满足 日 己 的 需要 。 


9.1.2 对象 方 法 


方法 为 对 象 添加 了 功能 ， 对 象 不 再 只 是 静 静 地 坐 着 ， 只 是 保存 属性 
而 任 岁月 跷 路， 它们 会 实际 地 做 一 些 事情 。 程 序 清单 9.4 展 示 了 这 一 


(©) 


程序 清单 9.4， 具有 方法 的 一 个 区 


: <?php 

: Class myClass { 

function sayHello{) { 
echo "HELLO!"; 


} 


} 
: $object1 = new myClass({); 
: $object1 -> sayHello(); 
?> 


Go” DO 上 PP 一 


尽管 这 不 是 方法 的 最 激动 人 心 的 例 于 ， 如 果 把 这 段 代码 保存 为 
helloclass.php， 将 其 放置 在 文档 根 目录 下 ， 使 用 Web 浏 览 需 访问 它 ， 将 
会 看 到 如 下 的 显示 结果 。 

HELLO! 

因此 ， 一 个 方法 的 外 观 及 其 行为 束 像 一 个 正常 的 函数 ， 只 不 过 方法 

是 定义 在 一 个 类 的 框架 中 的 。-> 操 作 从 用 来 在 脚本 中 调用 对 象 方 法 。 只 


要 关中 定义 了 变量 ， 方 法 都 能 够 访问 它们 以 供 目 己 使 用 。 程 序 清 单 9.5 
说 明了 这 一 后。 





程序 清单 9.5 在 一 个 方法 中 访问 类 属性 


<?php 
class myClass { 
public$name = "Jimbo"; 
function sayHello{) { 
echo "HELLO! My name is ".$this->name; 
} 
} 
$object1 = new myClass(); 
$objecti -> sayHello({); 
eo 


=. Oa nN DOoah WNhM = 
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如 果 把 这 段 代 人 码 保 存 为 helloclass.php， 将 其 放置 在 文档 根 目 录 下 ， 
使 用 Web 浏 览 占 访问 它 ， 将 会 看 到 如 下 的 显示 结 琳 。 


HELLO! My name is Jimbo 


正如 我 们 在 第 5 行 看 到 的 ， 特 殊 的 变量 $this 用 来 引用 当前 实例 化 的 
对 象 。 任 何 时 候 ， 如 果 一 个 对 象 引 用 自身 ， 都 必须 使 用 $this 变 量 。 把 
$this 变 量 和 -> 操作 符 结合 起 来 使 用 ， 我 们 就 能 够 在 类 自身 之 中 访问 其 任 
何 属性 或 方法 。 


有 关 对 象 属性 基础 用 法 的 最 后 一 个 小 例子 ， 束 是 如 何在 一 个 方法 中 
更 改 一 个 属性 。 而 在 前 面 ， 属 性 都 是 在 包含 它 的 方法 之 外 修改 。 程 序 清 
单 9.6 展 示 了 如 何 做 到 这 一 点 。 


程序 清单 9.6 在 一 个 方法 中 改变 一 个 属性 的 值 


1: <?php 
2: class myClass { 
3: public$name = "Jimbo"; 


4 function setName($n) { 

5i $this->name = $n; 

6: } 

了 function sayHello() { 

8: echo "HELLO! My name is ".$this->name; 
3 } 

1@: } 


11: $objecti = new myClass(); 

12: $objecti -> setName("Julie"); 
13: g$object1 -> sayHello(); 

14: ?> 


如 果 把 这 段 代 码 保存 为 helloclass3.php， 将 其 放置 在 文档 根 目 录 下 ， 
使 用 web 浏览 右 访 问 它 ， 将 会 看 到 如 下 的 显示 结 采 。 


HELLO! My name is Julie 


为 什么 ? 因为 在 第 4 一 6 行 中 ， 创 建 了 一 个 名 为 setName0 的 新 图 数 。 
当 在 第 12 行 调用 它 的 时 候 ， 它 把 $name 的 值 修 改 为 Julie。 因 此 ， 当 在 第 
13 行 调用 sayHello0 函 数 的 时 候 ， 它 三 找 $this->name， 它 使 用 了 Julie， 这 
是 刚刚 由 setName0O 国 数 设置 的 新 值 。 换 句 话 说 ， 一 个 对 象 能 够 修改 目 己 
的 属性 ， 在 这 个 例子 中 ， 束 是 $name 变 量 。 


9.1.3 ”构造 方法 


构造 方法 (constructor) 是 存在 于 类 中 的 一 个 函数 ， 它 和 类 同名 ， 
当 使 用 new classname 创 建 类 的 一 个 新 实例 的 时 候 ， 目 动 调 用 构造 方法 。 
使 用 构造 方法 ， 我 们 能 够 为 类 提供 参数 ， 当 类 调用 的 时 候 ， 参 数 束 会 江 
即 处 理 。 我 们 将 在 下 一 节 看 到 天 于 对 象 继 承 的 构造 方法 的 应 用 。 


9.2 xX} RAKAK 


学 习 了 了 对象、 属性 和 方法 的 基础 之 后 ， 我 们 可 以 开始 看 看 对 象 继 
际 。 和 类 相关 的 继承 是 名 副 其 实 的 ， 一 个 类 可 以 从 其 父 类 那里 继承 功 
能 。 程 序 清单 9.7 给 出 了 一 个 例子 。 


程序 清单 9.7 继承 其 父 类 的 一 个 类 


<?php 
class myClass 1 
public$name = "Matt"; 
function myClass($n) { 
$this->name = $n; 
} 
function sayHello{) { 
echo "HELLO! My name is ".$this->name; 


O NDOU BON — 


9: } 

10: } 

11: class childClass extends myClass { 

12: //code goes here 

ian 1 

14: $object1 = new childClass("Baby Matt"); 
15: $object1 -> sayHello{); 

1o: 7> 


如 果 把 这 段 代 码 保存 为 inheritance.php， 将 其 放置 在 文档 根 目 录 下 ， 
使 用 Web 浏 览 右 访问 它 ， 将 会 看 到 如 下 的 显示 结 采 。 


HELLO! My name is Baby Matt 


第 4 一 6 行 束 是 一 个 构 千 函数。 注意， 这 个 函数 的 名 字 和 包含 它 的 类 
的 名 字 相 同 ， 都 是 myClass。 第 11 一 13 行 又 定义 了 一 个 疾 ， 即 
childClass, ERAGE. Rep, AATEABIA, EMMA 
在 于 展示 从 父 关 的 继承 。 继 承 通 过 使 用 extends 子 句 来 肥 生 ， 如 第 11 行 所 
示 。 由 于 使 用 了 这 个 子 句 ， 第 二 个 类 继承 了 第 一 个 类 的 元 系 。 


Ba “BP RAN S FRU th ARN TI, BLE PR 
9.8. 


程序 清单 9.8 PRIA Bi [RANT 


: <?php 
class myClass { 

public$name = "Matt"; 

function myClass($n) { 
$this->name = $n; 

} 

function sayHello{) { 
echo "HELLO! My name is ".$this->name; 


ONO Of ON — 


ai } 

10: } 

11: class childClass extends myClass { 

T2: function sayHello({) { 

ta: echo "I will not tell you my name."; 
14: } 

To: } 

16: $object1 = new childClass( "Baby Matt"); 

17: $objecti -> sayHello{); 

18: ?> 


ARE PPS 9.7 CRSA EL, CB TSE — WY OE a FE TB 12 ~ 
14 行 。 在 这 些 行 中 ， 创 建 了 一 个 名 为 sayHello0 的 函数 ， 显 示 了 消息 < 
will not tell you my name”， 向 个 十 显示 “HELLO! My name is...”。 现 
在， 由 于 sayHelloO 函 数 存在 于 childClass 中 ， 和 而 childClass 古 第 16 行 所 调 
HWX, ALA, WEHE HsayHello( hx As . 


如 果 把 这 段 代 码 保存 为 inheritance2.php， 将 其 放置 在 文档 根 目录 
下 ， 使 用 Web 浏 览 器 访问 它 ， 将 会 看 到 如 下 的 显示 结果 。 
I will not tell you my name 

LR TE AI OT RIET IRER RSE SABENERE oe 
活 的 时 候 ， 继 承 是 有 用 的 。 假 设 你 创建 了 一 个 文本 格式 化 类 ， 它 组 织 和 
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拥有 了 个 人 的 杰作 。 现 在 ,假设 有 一 个 客 尸 想 要 使 用 这 一 想法 ， 但 是 他 
项 要 把 内 容 格 式 化 为 纯 文本 并 你 存 到 一 个 文本 文件 中 ， 而 不 是 把 内 容 格 
式 化 为 HTML 并 友 壕 给 浏 史上 禹 。 没 问题 ， 只 需要 添加 几 个 方法 和 属性 ， 

然后 你 就 完事 了 。 最 后 ， 那 位 客户 跑 来 说 他 其 实 想 要 把 数据 格式 化 并 作 
为 一 封 E-mail 友 运 。 这 是 怎么 搞 得 ， 为 什么 没有 创建 XML 格 式 化 文件 

WE ? 
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fifi FGA NERF. TRE HRE (HTML, WE, Email 
和 XML) 部 是 如 些 ， 你 束 基 本 拥有 了 一 个 父 类 - 子 类 关系 。 包 含 编译 和 
存储 方法 的 那个 是 父 类 。 负 贡 格 式 化 的 是 子 类 ， 它 们 从 父 类 继承 信息 并 
且 根 据 自己 的 功能 来 输出 结果 。 于 是 所 有 人 都 满意 了 。 


9.3 小结 
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面 同 对 象 程序 设计 的 方方面面 ， 大 学 里 已 经 有 教授 天 于 这 一 主题 的 一 系 
列 的 康 程 ， 因 此 ， 你 可 能 寻 得 这 里 的 遍 幅 是 有 点 少 。 然 而 ， 你 学 到 了 如 
何 创建 类 以 及 从 类 实例 化 对 象 。 你 学 习 了 如 何 创 建 和 访问 一 个 类 的 属性 
和 方法 ， 如 何 构 建新 的 关 ， 以 及 如 何 从 父 关 继承 功能 。 这 些 也 并 不 少 。 


9.4 Q&A 


Q: 为 什么 我 在 属性 声明 中 看 到 的 是 var 而 不 是 public、 


private=X protected. 


A: 在 PHP 的 较 早 版 本 中 ，var 用 来 在 类 中 声明 属性 。 为 了 同 后 兼 
容 ， 如 果 你 的 代码 中 仍然 使 用 var， 会 将 其 当 作 public 对 竺 而 不 会 引发 任 
何 问题 〈 除 非 你 想 要 它 成 为 private 或 protected 属 性 ) 。 


Q: 要 成 为 一 名 好 的 PHP 程 序 员 ， 或 者 说 只 是 要 读 完 本 书 ， 
我 人 须要 理解 面 同 对 象 程序 人 设计 吗 ? 


A: 并 非 如 此 。 实 际 上 ， 本 书 中 的 项 目 部 是 过 程式 的 而 不 包含 面 问 
对 象 的 程序 设计 。 面 问 对 象 程序 变 计 是 以 提高 组 成 一 个 给 定 应 用 程序 的 
代码 的 重用 性 和 扩展 性 为 目的 的 一 种 组 织 方法 。 在 一 个 面 同 对 象 设 计 有 的 
完整 规划 的 最 和 煞 阶 段 ， 你 可 能 对 项 目 知道 的 还 不 够 多 。 妆 它 完 成 后 ， 或 
者 至 少 达 到 一 个 稳定 的 状态 时 ， 你 才能 够 开始 看 看 在 哪些 领域 可 以 采取 
面 问 对象 的 方法 ， 并 且 可 以 开始 把 代码 组 织 成 关 、 属 性 和 方法 。 但 是 ， 
大 多 数 时 候 ， 执 行 特定 任务 的 价 单 脚本 并 没有 用 面 同 对 象 的 方式 去 纺 
Tj, RIERA DM A X RIET wit AA R o 


要 更 多 地 了 解 PHP 的 对 象 模型 ， 请 参见 PHP 手 册 的 相关 部 分 ， 
在 http:/www.php.net/ manual/en/language.oop5.php 可 以 找到 。 


9.5 KRAJ 


这 个 实践 练习 设计 用 来 帮助 你 预测 可 能 的 问题 、 复 习 已 经 学 过 的 知 
识 ， 并 且 开 始 把 知识 用 于 实践 。 


1. 如 何 声明 一 个 名 为 emptyClass 的 类 ， 它 没有 方法 也 没有 属性 ? 
2. 应 该 怎样 为 构造 方法 选取 名 字 ? 


3. 如 果 一 个 变量 声明 为 private， 那 么 ， 它 可 以 在 哪里 使 用 。 


oy 


fi E 


1. 使 用 dass 天 键 字 即 可 。 


class emptyClass { 


2， 你 没 法 选择 ， 构 造 方法 使 用 它 所 在 的 类 的 名 字 。 


3. 声明 为 private 的 变量 只 能 在 类 上 自身 中 使 用 。 


Fae jel 


1. 创建 一 个 名 为 baseCalc 的 类 ， 它 存储 了 两 个 数字 作为 属性 。 接 下 
来 ， 创 建 一 个 calculate0) 方 法 ， 访 方法 同 浏 览 需 最 示 数 字 。 


2. 为 思考 题 1 中 有 的 类 创建 名 为 addCalc()、subCalc(O)、mulCalcO 和 
divCalc() 的 方法 ， 它 们 窗 盖 了 calculate() 方 法 并 且 同 浏览 右 显 示 相 应 的 计 
FLAK 


第 3 部 分 “深入 编程 


使 用 字符 串 、 日 期 和 时 间 
使 用 表单 

使 用 Cookie 和 用 户 会 话 
使 用 文件 和 目录 
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第 10 草 ”使 用 字符 串 、 日 期 和 时 间 


在 本 章 中 ， 你 将 学 到 : 


如 何 格 式 化 字符 串 ; 

如 何 确定 一 个 字符 串 的 长 度 ; 

如 何在 一 个 字符 串 中 三 找 子 串 ; 

如 何 把 一 个 字符 串 分 解 为 组 成 部 分 ; 
如 何 从 一 个 字符 串 的 开头 和 结尾 删除 空格 ; 
如 何人 蔡 换 子 字 符 串 ; 

如 何 改 变 一 个 字符 串 的 大 小 写 ; 

如 何 获取 当前 日 期 和 时 间 ; 

如 果 获 取 有 关 日 期 和 时 间 的 信息 : 
如 何 格式 化 日 期 和 时 间 信 息 : 

如 何 验证 日 期 的 有 效 性 : 

如 何 设置 日 期 和 时 间 。 


不 官 Web 内 容 变 得 多 么 丰 宣 ， 其 背后 都 是 HTML 在 把 基于 字符 串 的 
内 容 显 示 到 Web 浏 览 右 。 因 此 ，PHP 提 供 了 很 多 函数 用 来 格式 化 和 操作 
字 从 串 ， 这 也 束 旱 不 令 人 意外 了 。 类 似 的 ， 日 期 和 时 间 是 每 天 生活 中 很 
重要 的 一 部 分 ， 以 全 于 我 们 不 必 多 想 也 很 容易 地 用 到 它们 。 然 而 ， 由 于 
公历 〈Gregorian calendar) 很 难 直 接 使 用 ，PHP 提 供 了 功能 强大 的 工 其 
使 得 对 日 期 的 操作 成 为 一 项 容易 的 任务 。 


有 很 多 的 PHP 函 数 可 供 使 用 ， 本 章 不 会 介绍 所 有 这 些 函 数 。 然 而 ， 
本 章 还 是 对 如 何 使 用 某 些 基本 的 字符 串 、 日 期 和 时 间 函 数 提供 了 一 个 基 
础 ， 并 且 针对 如 何 考 虑 在 代码 中 使 用 这 些 函 数 提供 了 一 个 想法 。 首 要 的 
原则 是 ， 如 果 你 想 要 转换 、 操 作 或 显示 一 个 字符 串 、 日 期 或 时 间 ， 在 参 
考 pPHP 手 册 之 前 ， 不 要 尝试 去 构建 自己 的 解决 方案 ， 因 为 ， 可 能 已 经 存 
在 一 个 函数 可 以 完成 你 想 要 完成 的 任务 。 


10.1 使 用 PHP 格 式 化 字符 串 


直到 目前 ， 我 们 接触 的 都 是 简单 地 以 其 了 最初 状 态 御 接 显 示 到 浏 席 关 
的 字符 串 。PHP 提 供 了 两 个 函数 ， 人 允许 我 们 格式 化 字符 串 ， 把 双 精 上 度数 
舍 入 到 指定 的 小 数位 数 ， 或 在 一 个 字段 中 定义 斜体， 或 根据 不 同 的 数值 
系统 来 显示 数据 。 在 本 节 中 ， 我 们 将 看 到 printfO 和 sprintf(0) 所 提供 的 一 些 
格式 化 选项 。 


10.1.1 使 用 printf() 


如 果 你 其 有 类 似 C 语 言 的 程序 设计 语言 的 经 验 ， 束 会 熟 炙 printf() 也 | 
数 的 概念 。printftO 函 数 需要 一 个 字符 串 作 为 参数 ， 叫 做 格式 控制 字符 串 
(format control string〉。 它 还 接受 附加 的 不 同类 型 的 参数 ， 我 们 将 稍 
后 和 学习。 格式 控制 字符 串 包含 了 与 显示 这 些 附 加 参数 相关 的 指令 。 例 
如 ， 如 下 的 代码 段 ， 使 用 printf0) 来 把 一 个 整数 输出 为 八进制 的 (或 者 以 
8 为 基数 的 ) 数 。 


<?php 

printf("This is my number: %o", 55); 
上 prints "This is my number: 67" 

o> 


= 格式 控制 字符 串 《〈 第 一 个 参数 ) 内 ， 我 们 包含 了 一 个 特殊 的 代 
代 ， 这 束 是 转换 指示 (conversion specification〉。 转 换 指 示 以 一 个 百 分 
写 (%) 打 头 ， 并 且 定 义 了 如 何 对待 printf() 相 应 的 参数 。 我 们 可 以 在 格式 
控制 学 符 串 中 包含 任意 多 个 转换 指示 ， 只 要 问 printf0 友 送 相 等 数目 的 参 
AUW AJ EA T o 
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<?php 

printft{ "First number: %f<br/>Second number: %f<br/>", 55, 66); 
// Prints: 

// First number: 55.000000 


// Second number: 66.000000 
?> 


第 一 个 转换 指示 对 应 printfO 的 第 一 个 附加 参数 ， 即 55， 第 二 个 转换 
指示 对 应 66。 跟 在 白 分 写 后 面 的 {要 求 把 数据 当 作 一 个 溯 点 数 ， 转 换 指 
示 的 这 个 部 分 叫做 类 型 指定 从 (type specifier) 。 


printf0 和 将 型 指定 人 符 


我 们 已 经 遇 到 了 两 种 类 型 指定 符 ， 以 八进制 显示 整数 的 o， 以 及 以 
点 数 显 示 整 数 的 f。 表 10-1 列 出 了 其 他 可 用 的 类 型 指定 符 。 


表 10-1 ”类 型 指定 符 


以 十 进 制 数 


以 二 进 制 数 显示 一 个 整数 
以 对 等 的 ASCII 显 示 一 个 整数 


以 浮 后 数 〈 双 精度 ) 显示 一 个 整数 


以 八进制 数 〈 以 8 为 基数 ) 显示 一 个 整数 





S 以 字符 串 显 示 参 数 


以 一 个 小 写 十 六 进 制 数 〈 以 16 为 基数 ) 显示 一 个 整数 





以 一 个 大 与 十 六 进 制 数 〈 以 16 为 基数 ) 显示 一 个 整数 





程序 清单 10.1 使 用 printfO 来 根据 表 10-1 列 出 的 某 个 类 型 指定 符 显 示 
一 个 时 个 的 数字 。 注 意 ， 我 们 不 仪 同 格式 控制 字符 串 添加 了 转换 指示 ， 
包含 其 中 的 任何 附加 文本 也 将 显示 出 来 。 


程序 清单 10.1 ”显示 某 些 类 型 指定 符 


1: <?php 

2: $number = 543; 

3: printf("Decimal: %d<br/>", $number); 

4: printf("Binary: %b<br/>", $number); 

5: printf("Double: %f<br/>", $number}; 

6: printf("Octal: %o<br/>", $number); 

7: printft("String: %s<br/>", $number); 

8: printf("Hex (lower): %x<br/>", $number); 
9: printft("Hex (upper): %X<br/>", $number); 
10: ?> 


把 上 述 代 码 放 到 一 个 名 为 printftestphp 的 文本 文件 中 ， 并 且 把 这 些 
文件 放 到 Web 服 务 器 文档 根 目录 下 。 当 通过 Web 浏 览 器 访问 这 些 脚本 的 
时 候 ， 会 产生 如 图 10-1 所 示 的 输出 。 正 如 你 所 看 到 的 ，printfO 是 一 种 把 
数据 从 一 种 数值 系统 转换 为 态 一 种 数值 系统 并 将 结果 输出 的 快捷 方法 。 





fa] localhost/10/printttest.php 
€ @ © http://localhost/10/printftest.php a 
Decimal: 543 
Binary: 1000011111 
Double: 543 000000 
Octal: 1037 
String: 543 
Hex (lower): 21f 
Hex (upper): 21F 


图 10-1 显示 转换 指示 


在 HTML 中 指定 一 种 颜色 的 时 候 ， 我 们 把 3 个 00 到 FF 之 间 的 十 六 进 
制 数 组 合 起 来 ， 它 们 分 别 表示 红 、 绿 和 蓝 。 我 们 可 以 使 用 printfO 把 0 到 
255 之 间 的 3 个 十 进 制 数 转换 为 相等 的 十 六 进 制 形式 ， 示 例如 下 。 


<?php 

$red = 204; 
$green = 204; 
$olue = 204; 


printf ("#%X%X%X", $red, $green, $blue}; 
ff prints "#CCCCCC" 
?> 


尺 省 我 们 可 以 使 用 类 型 指定 从 来 实现 从 十 进 制 到 十 六 进 制 的 数字 转 
换 ， 但 我 们 无 法 使 用 它 来 确定 每 个 参数 在 输出 中 占 多 少 个 字 人 和 从。 在 一 个 
HTML 颜色 代码 中 ， 每 个 十 六 进 制 数字 应 该 后 两 个 字符 ， 如 末 我 们 把 前 
面 代码 段 中 的 变量 $red、$green 和 $blue 的 值 改 为 1， 那 将 产生 问题 ， 得 


到 的 输出 结 末 是 #111。 可 以 通过 使 用 填充 指示 符 (padding specifier) 5 
制 输出 前 面 以 0 打头 。 


使 用 填充 指示 和 从 填充 输出 


我 们 可 能 需要 通过 一 个 占 位 符 来 填充 输出 。 填 元 指示 符 应 该 直接 跟 
在 百 分 号 后 甸 ， 这 个 百 分 写 古 一 个 转换 指示 的 开 尖 。 要 使 用 占 位 的 0 来 
填充 输出 ， 填 充 指示 符 应 该 由 一 个 0 以 及 跟 在 后 面 的 我 们 希望 输出 占用 
的 字符 个 数 来 组 成 。 如 果 输 出 所 占用 的 字符 少 于 这 个 总 数 ， 之 前 的 差 千 
将 会 用 0 填 元 ， 示 例如 下 。 


<?php 
printf("%@4d", 36); 
ji prints "9036" 

?> 


要 使 用 局 位 空格 来 项 多 输出 ， 需 充 指示 符 由 一 个 空格 以 及 跟 在 后 面 
的 表示 输出 应 该 占用 的 字符 数目 组 成 ， 示 例如 下 。 


<?php 

printf("% 4d", 36) 
f/f prints " 36" 
> 





Dl ih ae Ne AN “SHTML LAREN. a DAE f h Eb El CS <pre> Ap 1K 
强制 显示 空格 和 新 行 ， 示 例如 下 。 


echo "The spaces will be visible"; 


OR A BE 78 FE CPS A CAS, AY LME Fe header() R AR 2X 48 Content-Typetn 
头 ， 示 例如 下 。 


header({ "Content-Type: text/plain"); 





列 筷 了 了 ， 为 了 让 header0 函 数 如 期 地 工作 ， 我 们 不 得 加 浏览 右 发 送 
任何 输出 。 可 以 在 填充 指示 和 从 中 指定 空格 或 0 以 外 的 任何 字符 ， 只 雷 要 
用 一 个 单 引 号 后 面 跟着 你 想 用 的 字符 束 行 了 了， 示例 如 下 。 


<?php 

printf ("%'xdd", 36}; 
{i prints °"xx36" 

?> 


我 们 现在 有 了 完成 HIML 代 但 示例 所 珊 的 工具 。 在 此 之 前 ， 我 们 可 
以 把 3 个 数字 转换 为 十 六 进 制 数 ， 可 以 不 用 0 项 序 ， 但 不 能 用 品位 0 来 需 
充 十 六 进 制 什 ， 示 例如 下 。 


<?php 

$red = 1; 

$green = 1; 

Golue = 1; 

printf ("#s@2X%02X%@2X%", $red, $green, $blue}; 
ff prints "#010101" 

?> 


每 个 变量 都 作为 十 六 进 制 数 输出 。 如 果 输 出 占 位 少 于 两 位 ， 将 会 添 
加 0。 


10.1.2 jE- NTR mE 


我 们 可 以 指定 输出 应 该 占用 的 位 数 。 字 段 宽 度 指 示 符 〈field width 
specifier) 是 一 个 整数 ， 假 设 没有 定义 填充 指示 符 ， 它 应 该 放置 在 表示 
转换 指示 的 白 分 写 的 后 面 。 如 下 的 代 人 码 段 输出 4 个 项 目的 一 个 列表 ， 所 
有 有 的 这 些 部 在 一 个 20 位 的 字段 中 。 为 了 让 这 个 位 置 在 浏览 帮 中 可 见 ， 我 
们 把 所 有 的 输出 放置 a 到 一 个 pre 元 系 中 。 


<?php 

echo "<pre>"; 

printt("%*28s\n°, “Books"); 
printt("%*2@s\n", "CDs" ); 
printt('%*2as\n", "“DVDs"}; 
printf("*2@s\n", "Games") ; 
printt{"%20s\n", "“Magazines"}; 
echo "</pre>"; 

v> 


图 10-2 给 出 了 这 个 代码 段 的 输出 。 





[=] localhost/10/sample.php 
他 © | © http://localhost/10/sample.php a, 


Books 

CDs 

DVDs 
Games 
Magazines 


图 10-2 ”使 用 字段 宽度 指示 符 对 齐 
团 认 情况 下 ， 输 出 在 你 所 指定 的 字段 中 是 右 对 章 的 。 我 们 可 以 通过 
在 子 段 宽度 指示 从 前面 加 一 个 减 写 ， 从 而 让 其 左 对 齐 。 


printf("%-2@s\n", "Left aligned"); 


注意 ， 对 章 方式 应 用 到 输出 中 的 任何 数字 的 小 数 部 分 。 换 名 话说 ， 
当 使 用 右 对 齐 的 时 候 ， 只 有 双 精 度数 的 小 数 点 前 面 的 部 分 和 字段 宽度 的 


末尾 对 齐 。 
指定 精度 


如 条 四 要 以 序 点 数 格 式 输出 数据 ， 可 以 指定 目 己 想 要 对 数据 售 入 的 
精度 。 当 处 理 货币 a HIR, IRR AH. MRSA AN TT DIY ARLE TER 
型 指示 和 从 的 前 面 。 它 包含 一 个 扩 以 及 后 面 跟 看 的 想 要 舍 入 的 小 数位 数 。 
这 个 指示 和 从 只 对 带 有 类 型 指示 符 f 有 的 输出 中 的 数据 有 效 ， 示 例如 下 。 


<?php 

printt("%.2f", 5.333333}; 
LE prints °5.33° 

了 > 


提示 : 


在 C 语 言 中 ， 可 以 和 printf() 一 起 使 用 一 个 精度 指示 和 从 来 指定 小 数 输出 的 填充 。 这 个 精度 指 
示 和 从 对 于 PHP 中 的 小 数 输 出 没有 作用 。 使 用 这 个 填充 指示 符 将 添加 整数 前 的 占 位 0。 


转换 指示 : 复习 

表 10-2 列 出 了 可 以 组 成 一 个 轻 换 指示 的 指示 和 从， 按照 它们 可 能 出 现 
的 顺序 。 注 意 ， 很 难 同 时 使 用 一 个 填充 指示 符 和 一 个 字段 宽度 指示 符 ， 
可 以 选择 使 用 其 中 的 一 个 ， 但 不 能 同时 使 用 。 


表 10-2 转换 指示 的 组 成 部 分 





,一 | 确定 输出 应 该 占用 的 字符 数 ， 以 及 如 果 占 位 不 够 应 该 添加 
SAFC TE AN TT — '4' 


的 字符 


字段 览 度 指 示 


确定 输出 格 却 化 后 的 宽度 


侍 
Ta ass 
类 型 指示 从 | 确定 应 该 输出 的 数据 类 型 e 


程序 清单 10.2 使 用 printf0 来 输出 产品 和 价格 的 一 个 列表 。 





程序 清单 10.2 ”使 用 printfO 来 格式 化 产品 价格 的 一 个 列表 


1: <?php 

2: $products = array("Green armchair" => "222.4", 
a "Candlestick"=> "4", 

4: "Coffee table => "8@.6"); 
5: echo "<pre>"; 

6: printt("%-20s%20s\n", "Name", "“Price"); 

7: printt("%'-4@s\n", ""); 

8: foreach ($products as $key=>$val) { 

9: printf( "%-20s%20.2f\n", $key, $val ); 
10: } 

11: echo "</pre>"; 

12: ?> 


我 们 自 完 在 第 2 行 到 第 4 行 定义 包含 产品 名 和 定价 的 一 个 天 联 数 组 。 
在 第 5 行 显 示 出 pre 元 素 有 的 开始 标记 ， 以 使 浏 唤 右 可 以 识别 空格 和 换行 。 
在 第 6 行 对 printfO 的 第 一 次 调用 使 用 了 如 下 的 格式 控制 字符 串 。 
"'s -205%205\n" 

FS CPT EAT AB AEN 2 “Se HTN C“%-208”) 定义 了 一 个 20 个 
字符 的 宽度 指示 符 ， 而 且 输 出 是 左 对 齐 的 。 还 使 用 了 字符 串 类 型 指示 
人 符 。 第 二 个 转换 指示 《〈“9%20s”) 设置 了 一 个 右 对 齐 的 字段 宽度 。 这 个 


printfO 调 用 将 会 输出 我 们 的 字段 标题 。 


第 7 行 的 printf0 函 数 调用 显示 了 包含 “-” 字 符 的 一 行 ， 它 有 40 个 字符 
的 客 度 。 我 们 使 用 一 个 坦 充 指示 符 来 实现 这 一 上 操 ， 它 会 为 空 日 子 从 串 洪 
MER. 


E 9ÍT printf) KRZA H, r m BZ M foreachi& Jm — hs 
分 。 在 这 里 使 用 两 个 转换 指示 ， 第 一 个 (“%-20s”) 把 产品 名 显示 为 一 
个 字符 串 ， 它 在 20 个 字符 的 一 个 字段 内 左 对 齐 ; 而 第 二 个 《“%20.2f”) 
使 用 一 个 字段 宽度 指示 符 来 确 体 和 输出 右 对 齐 的 20 个 字符 的 一 个 字段 。 我 
们 还 使 用 了 一 个 精度 指示 符 来 确保 和 输出 舍 入 到 两 位 小 数 。 


把 上 述 代码 放 入 到 一 个 名 为 printftest2.php 的 文本 文件 中 ， 并 且 把 这 
些 文 件 放 入 到 Web 服 务 右 文档 根 目 孙 下 。 当 通过 Web 浏 览 右 访问 这 些 脚 
本 的 时 候 ， 会 产生 如 图 10-3 所 示 的 输出 。 





localhost/10/printttest2.ph 


€ C | © http://localhost/10/printftest2.php `q 
Name Price 
Green armchair 222.40 
Candlestick 4.00 
Coffee table 60.60 


图 10-3 ”使 用 printfO 格 式 化 产品 和 价格 
10.1.3 ”参数 交换 


假设 要 把 日 期 输出 到 浏 贤 融 。 我 们 把 日 期 存储 在 一 个 多 维 数 组 中 ， 
并 且 使 用 printf(0) 来 格式 化 输出 ， 示 例如 下 。 


<?php 

Sdates = array/( 
array('mon'=> 12, ‘mday'=-25, ‘year'=>2011), 
array('mon'=> 1, ‘mday'=>23, ‘year‘’'=>2012), 
array('mon'=> 10, ‘mday'=>29, ‘year'=>2@11) 


); 
$format = include{"local format.php"); 
foreach($dates as $date) { 
printt('$format’, S$date['mon'], $dcate[ mday ], Sdate[ ‘year']}; 
} 


?> 


FE HUE RAS BA, te arse RF BA OE A —~h 4 Wlocal_format.php 
的 包含 文件 。 假 设 这 个 文件 只 包含 如 下 内 容 。 
<?php 


return "%02d/%02d/%d<br/>"; 
?> 


我 们 的 输出 将 遵守 如 下 格式 mm/dd/yyyy。 


12/25/2911 
01/23/2012 
10/29/2011 


假设 现在 我 们 要 在 欧洲 的 一 个 站 点 安 装 我 们 的 脚本 ， 那 里 的 日 期 格 
式 通 和 常 是 日 期 显示 在 月 份 前 面 (dd/mm/yyyy) 。 假 设 核心 代码 已 经 写 定 
并 且 无 法 修改 ， 我 们 该 怎么 办 呢 ? 羡 运 的 是 ， 我 们 现在 可 以 改变 参数 在 
格 去 控 制 代 码 中 的 顺序 ， 只 需要 把 return 语 句 修 改 如 下 。 


return "%2\$@02d/%1\$62d/%3\$d<br/>"; 


可 以 在 每 个 转换 指示 的 白 分 写 后 和 面 插 入 我 们 所 想 要 的 参数 数目 ， 后 
面 跟 看 一 个 转 义 的 美元 符 写 ($)。 这 样 ， 在 代码 段 中 ， 可 以 要 求 先 显示 如 
下 第 二 个 参数 ， 


%2\$02d 
接着 是 第 一 个 参数 ， 
%1\$82d 
最 后 是 第 三 个 参数 ， 
%3\$d 


BT EN CAS) 2 AR a ee SB DR Ce UH h H H. 


25/12/2011 
23/01/2012 
29/10/2911 


10.1.4 存储 一 个 格式 化 字符 串 


printfO 函 数 把 数据 输出 到 浏览 项 ， 这 意味 独 结 果 不 能 供 脚 本 使 用 。 
然而 ， 我 们 可 以 使 用 sprintfO 孙 数 ， 这 个 函数 的 使 用 方法 和 printfO 一 样 ， 
只 不 过 它 返 回 一 个 可 以 存储 以 供 稍 后 使 用 的 字符 串 。 如 下 的 代码 段 使 用 
sprintf() 来 把 一 个 双 精 度数 舍 入 为 两 位 小 数 ， 并 把 结果 存储 到 $cash 中 。 


<?php 

$cash = sprintf("%.2f", 21.334454); 
echo "You have \$$cash to spend."; 

f} Prints "You have $21.33 to spend." 
?> 


我 们 专门 使 用 sprintf() 来 把 格式 化 的 数据 写 入 到 一 个 文件 中 ， 你 可 
以 调用 sprintfO 并 且 将 其 返回 值 赋予 一 个 变量 ， 可 以 用 fputs0 把 这 个 变量 
输出 到 一 个 文件 中 。 


10.2 T f#@PHP HIE 


FATED he S EHAE S. PAT BR A AIR ORR 
包括 用 户 输入 、 数 据 库 、 文 件 和 Web 页 面 。 在 开始 操作 来 自 外 部 源 的 数 
据 之 前 ， 季 第 需要 了 解 有 关 数 据 的 更 多 信息 。PHP 拓 供 了 很 多 函数 ， 我 
们 可 以 用 来 获取 有 关子 从 串 的 信息 。 


10.2.1 索引 字符 串 的 一 个 注意 事项 


我 们 常常 需要 使 用 和 字符 串 相 关 的 一 个 词 ,，“ 索 引 ”(index) 。 在 第 
8 章 中 讨论 数组 的 时 候 已 经 过 到 过 这 个 词 。 实 际 上 ， 字 符 串 和 数组 并 不 
是 想象 的 那样 差异 迎 然 。 我 们 可 以 把 字符 串 看 作 是 字符 的 一 个 数组 ， 
此 ， 我 们 可 以 像 访 问 数 组 的 元 系 那 样 ， 访 问 一 个 字符 串 的 单个 字符 ， 示 
例如 下 。 


<?php 

Stest = "phpcoder'; 

echo $test[@]; // prints "p" 
echo $test[4]; // prints "o" 
?> 


宇 符 束 像 是 数组 的 元 系 ， 也 有 一 个 从 0 开始 而 不 是 从 1 开始 的 索引 。 
当 我 们 讨论 一 个 字符 串 中 的 字符 的 位 置 或 索引 的 时 候 ， 记 住 上 面 这 一 点 
rE TR HEIN « 


10.2.2 ”使 用 strlen0 获 取 一 个 字符 串 的 长 度 


我 们 可 以 使 用 内 建 的 stmlen0 函 数 来 确定 一 个 字符 种 的 长 度 。 这 个 函 
数 需 要 一 个 字符 串 作 为 参数 ， 并 且 返 回 一 个 整数 ， 表 示 我 们 传递 给 它 的 


这 个 字符 串 中 的 字符 的 数目 。sttlen0 可 以 用 来 检查 用 户 输入 的 长 度 ， 就 
像 在 下 面 的 代码 段 中 ， 它 检测 一 个 会 员 代 号 以 确保 其 确实 是 4 位 数字 的 
长 度 。 
<?php 
$membership = "pAB7"; 
if (strlen($membership} == 4} { 

echo "<p>Thank you!</p>'"; 


} else 1 
echo ‘<p>Your membership number must be four characters long.</p>"; 


I 


?> 


只 有 当 $membership 变 量 所 保存 的 字符 串 的 长 度 确 实 为 4 位 的 时 候 ， 
用 户 才 会 得 到 感谢 信息 。 人 否则 ， 将 会 显示 一 条 出 钳 信 息 。 


10.2.3 ”使 用 strstr0 获 取 一 个 字符 串 的 子 串 


我 们 可 以 使 用 strstr0 函 数 来 测试 一 个 字符 串 是 否 存在 于 另 一 个 字符 
串 之 中 。 这 个 函数 需要 两 个 参数 ， 源 字符 串 和 我 们 想 要 在 其 中 查找 的 子 
字符 事 。 如 采 没 有 找到 子 字 符 串 ， 这 个 函数 返回 false， 人 否则 ， 它 返回 源 
字符 串 的 一 部 分 ， 这 部 分 以 子 字符 串 开 始 。 在 下 面 的 例子 中 ， 假 设 对 于 
那些 会 员 代码 中 包含 了 字符 串 AB 的 会 员 和 没有 包含 字符 串 AB 的 会 员 ， 
我 们 要 区 分 对 生 。 


<?php 
$membership = "pAB7"; 
if (strstr($membership, "AB"}} { 
echo "<p>Your membership expires soon!</p>"; 
} else { 
echo "“<p>Tnank you!</p> ; 
f 


?> 


HH 4 $membershipH EES SPAT AB, strstr( A BOR [al 
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相应 的 信息 : “Your membership expires soon!”。 但 如 果 我 们 用 来 查找 的 
源 字符 串 是 “pab7”， 那 会 有 友 生 什么 情况 ?由 于 strstr0 古 区 分 大 小 与 的 ， 
将 不 会 找到 AB。 最 初 的 f 语 句 的 测试 将 会 失败 ， 浏 贤 右 将 会 显示 上 默认 消 
(“Thank you!”)。 如 果 我 们 想 要 在 字符 串 中 查找 AB 或 ab， 必 须 使 用 
substrO RKR Ë strstr O ， 访 图 数 的 用 法 完全 相同 ， 但 是 它 的 三 找 不 区 分 
JI T 


10.2.4 ”使 用 strpos() 找 到 一 个 子 字 从 串 的 位 置 


strposO 国 数 告诉 我 们 在 一 个 大 字符 溃 中 是 仍 存 在 一 个 字符 串 ， 并 且 
告诉 我 们 其 位 置 。strpos() 函 数 需 要 两 个 参数 : WET A A RN TF 
从 串 。 这 个 冰 数 还 接受 一 个 可 选 的 第 三 个 参数 ， 这 是 一 个 整数 类 型 的 
值 ， 表 示 要 从 哪里 开始 查找 的 索引 。 如 果 这 个 子 字符 串 不 存在 ，strpos() 
的 返回 值 为 false， 人 奋 则 ， 它 返回 子 凶 人 符 串 开始 的 索引 。 下 面 的 代 人 码 段 使 
用 strpos0) 来 确保 一 个 字符 串 以 字符 串 mz 开 始 。 


<?ohp 

$membership = "mz@@xyz" ; 

if (strpos($membership, "mz") === 0) 1 
echo "Hello mz!"; 


I 


?> 


注意 我 们 用 来 获得 期 望 的 结果 的 技巧 。 尽 管 strpos0O 函 数 在 我 们 的 字 
侍 串 中 三 找 mz， 它 从 第 一 个 元 系 开 始 得 找 ， 也 束 是 0 位 置 。 如 果 strpos() 
函数 的 返回 值 为 false， 那 么 在 if 条 件 测 试 中 ，false 也 将 返回 9。 为 了 解决 
这 一 点 ， 我 们 使 用 了 等 同 运 算 和 从 ===， 如 有 果 左 边 操 作 数 和 右边 的 操作 数 
相等 并 且 有 具有 相同 的 芙 型 ， 残 像 在 本 例 中 的 情况 一 样 ， 才 会 返回 true。 


这 只 是 用 来 在 haystacks 中 查找 needles 的 一 些 和 字符 串 相 关 函 数 的 几 
个 变 体 之 一 。 参 考 PHP 手 册 中 所 提供 的 关于 这 个 函数 的 内 容 ， 还 可 以 找 
到 其 他 相关 的 函数 。 


10.2.5 ”使 用 substr0 提 取 一 个 字 从 串 的 一 部 分 


substrO 函 数 根 据 开 始 索引 和 我 们 要 得 找 的 字符 串 的 长 度 来 返回 一 个 
字符 第。 这 个 函数 需要 两 个 参数 : 源 字 从 时 和 开始 索引 。 使 用 这 些 参 
数 ， 函 数 会 返回 从 开始 索引 到 我 们 所 僵 找 的 字符 串 的 末尾 的 所 有 入 从。 
我 们 也 可 以 《可 选 地 ) fete TBR, RAN BORIS BIR 
整数 。 如 果 使 用 了 第 三 个 参数 ，substr0 只 是 从 开始 索引 向 前 ， 返 回 指定 
数目 的 字符 。 


<?php 


$test = "phpcoder"; 

echo substr($test,3)."<br/>"; // prints "coder" 
echo substr($test,3,2)."<br/>"; // prints "co" 
?> 


如 有 果 传 递 给 substr0 一 个 负数 作为 第 二 个 参数 (开始 索引 ) ， 它 将 会 
从 字符 串 的 末尾 而 不 是 开头 开始 计算 。 和 下面 的 代码 为 那些 已 经 提交 了 一 
个 以 .ff 结尾 的 Email 地 址 的 人 编写 了 一 条 特定 的 消息 。 


<?php 
$test = "pierre@wanadoo.fr' ; 
if ($test = substr($test, -3) == ".fr"j) 1 


echo "<p>Bonjour! Nous avons des prix speciaux de vous.</p>"; 
} else { 

echo "<p>Welcaome to our store.</p>"; 
f 


?> 


10.2.6 ”使 用 strtok() 分 解 一 个 字符 串 


我 们 使 用 strtokO 函 数 一 个 单词 一 个 单词 地 解析 一 个 字符 串 。 这 个 范 


BN Fig SEPA BE rn BET EN AF DR OR oP BET BO AF. Ap 
aiaia 并 且 这 个 函数 将 返回 所 找到 的 第 一 
个 分 解 。 在 第 一 次 调用 strtokO 郴 数 后 ， 源 字符 串 将 放 入 缓存 ， 对 于 后 续 

的 调用 ， 只 需要 给 strtok0 函 数 传递 分 隔 符 字 符 串 。 每 次 调用 这 个 a 

时 ， 它 将 会 返回 所 找到 的 下 一 个 分 解 ， 当 到 达 字 符 串 末尾 的 时 候 返 

false。strtokO 函 数 通 闻 都 在 一 个 循环 中 重复 地 调用 。 scien 

ee 六 解 一 个 URL， 首先 从 碍 询 字 人 符 串 中 分 解 出 主机 和 路 径 ， 并 进 

步 分 解 出 得 询 字 符 串 的 名 / 值 对 。 


程序 清单 10.3 ”使 用 strtokO 〇 分 解 一 个 字符 串 


: <?php 
: $test = "http://www.google.com/search?"; 
: $test .= "hl=en&ie=UTF-8&q=php+development+books&btnG=Google+Search"; 
: $delims = "?7&"; 
$word = strtok($test, $delims}; 
: while (is_string($word)) { 
if ($word) { 
echo $word."<br/>"; 


O N a ahr ON 一 


co 


} 
10: $word = strtok($delims) ; 


把 上 述 代 码 放 入 到 一 个 名 为 teststrtotok.php 的 文本 文件 中 ， re 
这 些 文 件 放 入 到 Web 服 务 右 文档 根 目 录 下 。 当 通过 Web 浏 贤 右 访问 这 
脚本 的 时 候 ， 会 产生 如 图 10-4 所 示 的 输出 。 


[5] localhost/L0/teststrotok.ph 


= C © htto://localhost/10/teststrotok.ohp a, 


http2'www.google.com’search 
hi—en 

ie=LUTF-8 
qg=php+development+books 
binG=Google+Search 


110-4 ”teststrtotok.php 的 输出 ， 一 个 分 解 后 的 字符 串 


strtok0) 孙 数 像 一 个 有 后 机 械 的 指令 ， 使 用 它 需 要 一 些 技 巧 。 在 第 4 
行 ， 我 们 站 先 把 想 要 使 用 的 分 隔 稚 存储 a 到 一 个 变量 中 ， 即 $delims。 在 第 
5 行 调 用 strtok0， 把 想 要 分 解 的 UREL 以 及 $delims 字 符 串 作为 参数 传递 给 
它 ， 我 们 把 第 一 个 结果 存储 到 $word 中 。 在 第 6 行 的 while 循 环 的 条 件 测 
斌 中， 我 们 测 弃 $word 是 个 是 一 个 字符 串 。 如 末 个 是 ， 便 可 知道 已 经 到 
达 字 符 串 的 末尾 ， 并 且 不 再 需要 进行 任何 操作 。 


我 们 测试 返回 类 型 的 原因 ， 是 因为 这 样 一 种 情况 ， 一 个 字符 串 在 一 
行 中 包含 两 个 分 隔 符 ， 当 允 到 这 两 个 分 隔 符 中 的 第 一 个 的 时 候 ， 会 导致 
strtok0O 返 回 一 个 空 字符 串 。 因 此 ， 再 次 进行 下 面 的 测试 。 


while ($word) { 
Sword = strtok(S$delims}: 
} 


MR wode “N48 FAT FEAR, BEA A BAUS AT 
HY AS FE o 


既然 已 经 确定 了 $word 包含 一 个 字 从 串 ， 我 们 可 以 继续 使 用 它 。 如 
果 $word 没 有 包含 一 个 空 字符 串 ， 在 第 8 行将 其 显示 到 浏览 器 。 随 后 ， 必 
须 在 第 10 行 再 次 调用 strtok0 为 下 一 次 测试 重新 填充 $word 变量 。 注 意 ， 
我 们 不 要 再 次 把 源 字符 串 传 递 给 strtok()， 如 果 这 么 做 ， 源 字符 串 的 第 一 
个 单词 将 会 再 次 返回 ， 这 将 会 进入 一 个 无 限 循 环 。 


10.3 *EPHP Hee fy 


PHP Het SIRS BL, ALVA RIG — PT FT BBR, AVENE TE 
AEE EERE BOTA RG BIE 1 


10.3.1 使 用 trim0、ltrim0 和 strip_tags0 整 理 一 个 字符 串 


当 用 户 输入 或 者 从 一 个 外 部 文件 获取 文本 的 时 候 ， 我 们 总 是 无 法 确 
定 是 合 在 数据 的 开头 和 结尾 插 取 了 衬 日 字符 。trim0O 困 数 从 一 个 字符 证 
的 开头 和 来 尾 剔除 了 任何 空 晶 字符， 包括 换行 待 、Tab 符 和 衬 格 。 它 接 
受 要 修改 的 字符 串 作为 参数 ， 人 返回 一 个 整理 后 的 版 本 ， 示 例如 下 。 


<?php 

$text = "\t\tlots of room to breathe aie 

echo "<pre>$text</pre>" ; 

// prints " lots of raom to breathe E 
$text = trim($text): 

echo "<pre>$text</pre>"; 

// prints "lots of room to breathe"; 

?> 


当然 ， 这 可 能 不 是 我 们 所 期 望 的 工作 。 我 们 可 能 硕 望 傈 留 一 个 字符 
串 开 头 的 空白 而 删除 其 末尾 的 空白 。 我 们 可 以 使 用 PHP 的 rtrimO 函 数 来 
完成 和 使 用 trimO 所 做 的 同样 的 事情 。 但 是 ， 只 有 字符 串 末 尾 的 空 日 极 
删除 反 了 ， 示 例如 下 。 


<?php 

$text = "\t\tlots of room to breathe cae 

echo "<pre>$text</pre>"; 

{i prints " lots of room to breathe “a 
$text = rtrim($text}; 

echo "<pre>$text</pre>"; 

// prints ° lots of room to breathe"; 

?> 


PHP 提 供 了 ltrim0O 函 数 ， 它 只 从 一 个 字符 串 的 开头 删除 空 日 。 同 
样 ， 它 也 使 用 我 们 所 要 转换 的 字符 串 来 调用 ， 并 且 返 回 一 个 新 的 字符 
串 ， 其 中 删除 了 Tab 键 、 换 行 和 空白 ， 示 例如 下 。 


<?php 

$text = "\t\tlots of room to breathe g 

echo "<pre>S$text</pre>"; 

{i prints " lots of room to breathe oa 
$text = ltrim($text}; 

echo "<pre>$text</pre>"; 

ff prints "lots of room to breathe E 

?> 


从 一 段 文本 中 删除 标记 ， 从 而 不 市 HIML 格 式 地 显示 字符 串 ， 这 是 
很 常见 的 需求 。PHP 提 供 了 strip_tags() eh BOK IA BNE AY. strip_tags() ei 
数 接受 两 个 参数 : 要 转换 的 文本 和 一 个 可 选 的 strip_tagsO 可 以 保留 的 
HTML 标 记 的 集合 。 这 个 列表 中 的 标记 不 应 该 用 任何 字符 隅 开 ， 示 例如 
Fa 


<?php 

$string = "<p>\"I <em>simply</em> will not have it,\" <br/>said Mr Dean.</p> 
<o><strong>The end.</strong></p>": 

echo strip _tags($string, "<br/><p>"); 

?> 


在 上 面 的 代码 段 中 ， 我 们 创建 了 一 个 HIML 和 格式 的 字符 串 。 当 调用 
strip_tagsO 的 时 候 ， 台 把 $string 变 量 和 一 个 例外 的 标记 列表 传递 给 它 。 
结果 是 <p> 和 <br> 标 记 保留 ， 而 所 有 其 他 的 标记 都 删除 挥 。 注 意 ，<p> 
的 配套 标记 </p> 也 删除 了 。 


这 个 代码 段 的 输出 如 下 。 


"I simply will not have it," 
Said Mr Dean, 


The end. 


注意 ， 和 斜体 和 粗 体 的 格式 没有 了 ， 但 段落 和 换行 依然 保留 。 
10.3.2 ”使 用 substr_replace0 符 换 一 个 字符 串 的 一 部 分 


substr_replaceO 函 数 和 substrO 函 数 的 功能 美 似 ， 只 是 它 人 允许 我 们 把 
所 提取 出 的 那 部 分 字符 串 痊 换 反 。 这 个 图 数 需要 3 个 参数 : 要 修改 的 字 
人 符 串 、 需 要 添加 给 它 的 文本 以 及 开始 索引 。 它 还 接 党 一 个 可 选 的 长 度 参 
Šlo substr_replace() KRETI BFR I aa Re | KREBS AP aE 
的 部 分 ， 用 所 提供 的 字符 串 蔡 换 这 一 部 分 ， 然 后 返回 整个 修改 后 的 字符 
E o 


下 面 的 代码 段 用 来 更 新 一 个 用 户 的 会 员 代 但 ， 我 们 修改 了 其 中 第 三 
个 和 第 四 个 字符 。 
<?php 
Smembership = "mz1i1xyz"; 
$membership = substr_replace($membership, "12", 2, 2); 
echo "New membership number: Smembership'; 


// prints "New membership number: mz12xyz' 
?> 


这 段 代 人 码 的 结果 是 : 旧 的 会 员 吕 但 “mz11xyz” 被 转换 为 新 的 会 员 号 
f“mz12xyz” - 


10.3.3 ”使 用 str_replace(0 蔡 换 子 字符 串 


str_replace0O 函 数 用 来 把 另 一 个 字符 串 中 的 给 定 字 符 串 的 所 有 实例 都 
蔡 换 挥 。 它 需要 3 个 参数 : ERFIR, PRCA AE SAGER 
这 个 函数 返回 修改 后 的 字符 串 。 


如 下 的 例子 使 用 str_replaceO) 把 主人 字符 串 中 所 有 出 现 2010 的 地 方 修改 


42012. 


<?php 

$string = "<hi>The 2010 Guide to All Things Good in the World</h1>"; 
$string .= "<p>Site contents copyright 26190.</p>"; 

echo str_replace("2010","2012" ,$string}; 

了 > 


str_replace0O 函 数 的 所 有 参数 都 接受 数组 和 字符 串 的 形式 的 什 ， 这 就 
允许 我 们 对 一 个 主 字 人 符 串 甚 全 在 多 个 主 字 符 果 上 执行 多 次 租 找 和 蔡 换 操 
作 。 以 下 面 的 代码 段 为 例 。 
<?php 
$source = array[{ 

"The package which is at version 4.2 was released in 2005.", 


"The year 2005 was an excellent time for PointyThing 4.2!"}3; 
esearch = array{"4,2", "2005"); 


$replace = array({"6.3", "2012"); 
$source = str_replace($search, $replace, $source}; 
foreach{$source as $str) { 

echo "$str<br="; 


} 


?> 


这 段 代码 的 输出 如 下 。 


The package which is at version 6.3 was released in 2012. 
The year 2012 was an excellent time for PointyThing 6.3! 


当 一 个 字符 串 数 组 传递 给 str_replace0 作 为 第 一 个 和 第 二 个 参数 的 时 
候 ， 筷 会 答 试 在 要 修改 的 文本 中 ， 把 每 个 得 找 字 符 串 和 相应 的 蔡 换 字符 
串 作 交换 。 如 果 第 三 个 参数 是 一 个 数组 ，str_replace0 孙 数 返回 一 个 字符 
串 数 组 。 在 返回 的 字符 串 数 组 中 的 每 个 字符 串 上 已 经 执行 了 租 找 和 答 换 


10.3.4 ”转换 大 小 与 


PHPHeHe SJL PRA, TOMER ES BN ADS EK) SS 
第 对 字符 串 比 较 有 用 。 要 得 到 一 个 字符 串 的 大 写 厂 本 ， 使 用 函数 
strtoupper()。 这 个 函数 只 需要 一 个 想 要 转换 的 字符 串 ， 并 且 返 回转 换 后 
的 字符 串 ， 示 例如 下 。 


<?php 

$membership = "mzi1xyz"; 

Gmembership = strtoupper{$mnembership); 
echo "Smembership"; // prints "“MZ11XYZ" 
?> 


要 把 一 个 字符 串 转 换 为 小 号 字符 ， 使 用 strtolower0O 图 数 。 同 样 ， 
个 函数 只 需要 一 个 想 要 转换 的 字符 串 ， 并 且 返 回转 换 后 的 字符 串 ， 
如 下 。 


<?php 

S$membership = "MZ11XYZ"; 

Gmembership = strtolower({$mnembership); 
echo "S$membership"; // prints "“mz11xyz" 
?> 


PHP 还 提供 了 一 个 具有 很 好 的 痛 饰 作用 的 大 小 写 图 数 。ucwords0) 函 
数 使 得 字符 串 中 的 每 个 单词 的 第 一 个 字母 变 为 大 号 。 在 下 面 的 例子 中 ， 
我 们 将 使 得 用 户 提 区 的 字符 串 中 的 每 个 单词 的 第 一 个 字母 变 为 大 写 。 


<?php 
$full name = "violet elizabeth bott"; 


$full name ucwords($full name); 
echo $full name; // prints "Violet Elizabeth Bott" 
了 > 


尽管 这 个 函数 使 得 每 个 单词 的 第 一 个 字母 变 为 大 与 ， 但 它 并 没有 动 
其 他 的 字母 。 因 此 ， 在 前 面 的 例子 中 ， 如 果 用 户 的 Shift 键 有 问题 ， 并 且 
提交 了 VIolEt eLIZaBeTH bOTt，ucwords() 函 数 对 于 修正 字符 串 不 会 做 
太 多 的 工作 。 我 们 将 最 终 得 到 VIolEt ELIZaBeTHBOTt， 这 谈 不 上 有 太 


KAE. BOTT Ay UATE vid Fd ucwords()-Z Hija Wstrtolower() R BFE GE 20 WY 
字 从 串 转 换 为 小 写 ， 从 而 解决 这 个 问题 ， 示 例如 下 。 


<?php 

$full name = "VIOLEt eLIZaBeTH bOTt'"; 

$full name = ucwords(strtolower({$full name)}; 

echo $full_name; // prints "Violet Elizabeth Bott" 
?> 


最 后 ，ucfirstO 函 数 只 是 把 一 个 字符 串 的 第 一 个 字母 变 为 大 写 。 在 
下 面 的 代码 段 中 ， 我 们 把 一 个 用 户 提 交 的 字符 串 的 首 字 母 大 写 ， 示 例如 


<?php 

$myString = "this is my string."; 

$myString = ucfirst({$myString); 

echo $myString; // prints "This is my string." 
?> 


使 用 大 小 写 相 关 的 字符 串 函 数 是 有 用 的 ， 例 如 在 试图 验证 区 分 大 小 
写 的 密码 时 。 如 果 用 户 输入 “MyPass”， 而 存储 的 密码 是 “mypass”， 但 我 
们 希望 这 个 匹配 是 不 区 分 大 小 写 的 ， 可 以 答 试 将 用 户 输 入 的 小 写 或 大 写 
版 本 与 存储 的 密码 的 小 写 或 大 写 版 本 进行 比较 。 如 果 他 们 在 同样 大 小 瑟 
格式 下 是 匹配 的 ， 用 户 可 以 通过 验证 ， 即 使 他 输入 的 内 容 和 实际 存储 的 
内 容 有 一 些 不 同 。 


10.3.5 “使 用 wordwrap(0 和 mnl2br0 换 行文 本 


当 在 一 个 Web 页 面 中 最 示 纯 文本 的 时 候 ， 第 营 会 面临 一 个 问题 ， 换 
行 没 有 显示 并 且 文 本 乱 成 一 团 。nl2br0 函 数 方便 地 把 每 个 换行 符 转 换 为 
一 个 HTML 分 段 ， 示 例如 下 。 


<?php 


$string = "one linen"; 

$string .= "another linen"; 
$string .= ‘a third for luck\n'; 
echo nl2br($string) ; 

?> 


上 述 代码 输出 结果 如 下 。 


one line<br /> 
another line<br /> 
a third for luck<br /> 


nl2br0O 函 数 只 是 对 于 保持 要 转换 的 文本 中 已 经 存在 的 换行 符 效 果 很 
好 。 有 时 候 ， 我 们 可 能 需要 这 加 任意 的 换行 符 来 格式 化 一 串 文本 ， 
Wordwrap() 函 数 是 完成 这 一 任务 的 最 好 选择 。wordwrap0 需 要 一 个 参 
数 ， 即 要 转换 的 字 人 符 串 。 默 认 情 况 下 ，wordwrap0 函 数 把 每 75 个 字符 分 
为 一 行 ， 并 且 使 用 作为 换行 符 。 因 此 ， 示 例 代 但 段 如 下 。 


<?php 
$string = "Given a long line, wordwrap() is useful as a means of "; 
$string .= "breaking it into a column and thereby making it easier to read"; 


echo wordwrap($string); 
o> 


上 述 代 人 码 段 将 输出 如 下 的 结 来 。 


Given a long Line, wordwrap(} is useful as a means of breaking it into a 
column and thereby making it easier to read 


HA a ey A TI, RE SEAN HTMLA RÆ she 
示 。 当然 ，wordwrap0 〇 函数 还 有 两 个 可 选 参 数 : 表示 每 行 最 大 字符 数 的 
一 个 数字 以 及 要 用 来 表示 一 行 结 束 的 字符 串 。 因 此 ， 如 下 的 函数 调用 


echo wordwrap{$string, 24, "<br/>\n"}; 


应 用 到 我 们 的 $string 变 量 ， 输 出 结果 如 下 。 


Given a long line,<br/> 
wordwrap{) 18 useful as<br/> 
a means of breaking it<br/> 
into a column and<br/> 
thereby making it easier<br/> 
to read 


如 条 有 一 个 单词 的 字符 超出 了 限制 ，wordwrapO 函 数 并 不 会 在 行 限 
制 的 地 方 目 动 新 行 。 然 而 ， 我 们 可 以 使 用 第 四 个 可 选 的 参数 来 强制 实现 
这 一 氮 ， 这 个 参数 应 该 是 一 个 正 整数 。 因 此 ， 使 用 市 有 四 个 参数 的 
wordwrap0 〇 函数 ， 我 们 就 可 以 换行 一 个 字符 串 ， 即 便 是 它 所 包含 的 日 词 
超过 了 我 们 设置 的 限制 。 示 例 代 人 码 段 如 下 。 


<?php 

$string = "As usual you will find me at http://www.witteringonaboutit.com/"; 
$string .= "chat/eating green cheese/forum.php. Hope to see you there!"; 

echo wordwrap($string, 24, "<br/>\n", 1); 

?> 


上 述 代码 段 将 输出 如 下 结 来 。 


As usual you will find<br/> 
me at<br/> 

http://www. witteringonab<br/> 
outit.com/chat/feating gr<br/> 
een cheese/forum.php.<br/> 
Hope to see you there! 


上 述 代码 段 不 会 输出 如 下 结 末 。 
As usual you will find<br/> 
me at<br/> 


http: //www.witteringonaboutit.com/chatfeating green cheese/forum.php. <br/> 
Hope to see you there! 


10.3.6 1$ H explode) E FIF 4) fr PATH 


explode() K% Mstrtok0 EAI RRE IKMN. (Ate explodeQ HAGE 
一 个 字符 串 分 解 到 一 个 数组 中 ， 然 后 ， 我 们 可 以 按照 上 自己 的 意愿 去 存 


a a a a a 
BAT AU BWR AT AN Yo AARAA EH PE = SSR, SS 
hint 
个 字符 ， 所 有 这 些 字 符 构 成 一 个 单个 的 分 隔 符 《〈 不 像 是 传递 给 strtokO) 的 
多 个 分 隅 字符 ， 其 中 每 个 分 隔 字符 目 成 一 体 ) 。 下 面 的 代码 段 分 隔 一 个 
日 期 并 且 把 结 朱 存储 到 一 个 数组 中 。 


<?php 

$start_date = "2012-02-19"; 

$date array = explode("-", $start date}; 

{i $date_array[®] == "2012" 

f/f $date array[1] == "@2" 

fi $date_array[2] == "19" 

echo $date array[@]."-".$date array[1]."-".$date array[2]; 
ffprints 2012-02-19 

?> 


ME, RRE ECAP A PHPF E R T 
REIT 2 H RHAI HY TA) BR Z o 


10.4 1H PHP HY EBA AY TE] ek i 


下 面 的 小 市 介绍 PHP 中 与 日 期 和 时 间 相 关 的 函数 。 目 己 竹 试 列表 中 
的 每 个 函数 ， 可 以 及 现 这 些 函 数 是 如 此 简单 而 且 功 能 强大 。 


10.4.1 ”使 用 time0) 获 取 日 期 


PHPHtime(O) 函 数 提供 了 所 有 策 要 知道 的 天 于 当前 日 期 和 时 间 的 信 
恩 。 它 不 需要 参数 ， 并 且 返 回 一 个 整数 。 对 于 我 们 人 类 而 谨 ， 返 回 的 数 
字 有 所 难以 识别 ， 但 无 论 如 何 ， 它 特别 有 用 。 


echo time({); 
// sample output: 1326853185 
{i this represents January 17, 2012 at @9:19PM 


time0 返 回 的 整数 表示 从 GMT 1970 年 1 月 1 日 午夜 开始 流逝 过 的 秒 
数 。 这 个 时 刻 叫 做 UNIX 时 间 惟 CUNIX Epoch，〉， 从 那 时 起 已 流逝 的 秒 
数 就 叫做 时 间 惟 〈time stamp) 。PHP 提 供 了 优秀 的 工具 来 把 一 个 时 间 
稚 转 换 为 人 类 所 习惯 的 形式 。 即 便 如 此 ， 你 可 能 还 是 会 问 , “一 个 时 间 
愉 难 道 不 是 存储 日 期 的 一 种 无 用 而 费解 的 方式 吗 ? ”实际 上 恰恰 相反 。 
仅仅 从 一 个 数字 ， 我 们 就 可 以 提取 出 很 多 的 信息 。 更 何况 时 间 礁 使 得 日 
期 的 计算 比 我 们 想象 的 更 加 容易 。 


设想 一 个 目 制 的 日 期 系统 ， 其 中 我 们 记录 了 一 个 月 的 第 几 天 以 及 月 
份 和 年 份 。 现 在 ， 假 设 一 个 脚本 必须 为 一 个 给 定 的 日 期 添加 一 天 ， 如 果 
这 个 日 期 正好 是 1999 年 12 月 31 日 ， 我 们 必须 编写 代码 把 当月 的 天 数 设 置 
为 1， 把 月 份 设置 为 1 月 ， 把 年 份 设置 为 2000， 而 不 是 在 原来 的 日 期 上 加 
1。 使 用 一 个 时 间 玲 ， 我 们 只 需要 在 当前 的 数值 上 添加 一 天 所 包含 的 秒 


A (60x60x24, BY86400) , SWERK STA. ERITA EWA MR, 
可 以 把 这 个 新 数字 转换 为 菜 种 更 友好 的 形式 。 


10.4.2 ”使 用 getdate(0) 转 换 一 个 时 间 玲 


现在 我 们 可 以 使 用 一 个 时 间 惟 了， 在 将 其 展示 给 用 户 之 前 ， 我 们 必 
AUTRE o getdate() es AU FY ehh ese —-P A TE, JF ARES SAK 
日 期 信息 的 一 个 关联 数组 。 如 末 省 略 了 时 间 崔 ，getdate(O) 使 用 timeO 所 返 
加 的 当前 时 间 崔 。 和 10-3 列 出 了 包 作 在 getdateO0 所 返回 的 数组 中 的 元 
系 。 


表 10-3 ”getdate0) 所 返回 的 关联 数组 


一 月 中 的 天 数 (1 一 31) 
m pme p 


year 份 〈4 位 数字 ) 2008 





| | l i | l | -一 


一 年 中 的 天 数 (0~365) 


ERY Te) ax 1092065443 


程序 清单 10.4 在 第 2 行使 用 getdate0) 来 从 一 个 时 间 惟 中 提取 信息 ， 在 
第 3 行 用 一 条 foreach 语 句 来 显示 每 个 元 际 。 我 们 可 以 看 到 图 10-5 所 示 的 
典型 输出 。getdateO 国 数 根据 本 地 时 区 返回 日 期 。 








[a] localhost/10/getdate.php 


© C | hitp://localhost/10/getdate,php a, 


seconds = 11 
minutes = 2] 

hours = 21 

mday = 17 

wday = 2 

mon = | 

year = 2012 

yday = 16 

weekday = Tuesday 
month = January 


0 = 1326853271 


A A 


Today's date: 1/17/2012 


110-5 ”使 用 getdate() 


程序 清单 10.4 ”使 用 getdate() 获 取 日 期 信息 


1: <?php 

2: $date array = getdate({); // no argument passed so today's date will be used 
3: foreach ($date array as $key => $val) { 

4: echo "$key = $val<br>"; 

D: F 

6: ?> 

re <hr> 

8: <?php 

9: echo "<p>Today's date: ".$date array['mon']."/".$date array['mday']."/". 
10: $date array['year']."</p>"; 

117 2> 


BRIS TBS, Ras BCU PARRE. 


Warning: getdate{): It is not safe to rely on the system's timezone settings. 
You are *required* to use the date.timezone setting or the 
date default timezone set() function. 


你 需要 对 php.ini 文 件 做 一 些 修改 。 特 别 是 ， 找 到 如 下 的 内 容 的 位 
Be 


; Defines the default timezone used by the date functions 
; http: //php.net/date.timezone 
;cdate,.timezone = 


将 date.timezone 的 值 修改 为 你 的 时 区 《参见 
http://php.net/date.timezone 以 获取 有 效 值 的 一 个 列表 ) ， 示 例如 下 。 


; Defines the default timezone used by the date functions 
; http: //php.net/date.timezone 
date.timezone = America/New York 


确保 在 对 php.ini 做 出 修改 后 重新 局 动 Apache。 
10.4.3 ”使 用 date0 转 换 一 个 时 间 戳 


RATE E H es aan cH A ues I fie, a LAE H getdate() es 
出 但 是 ， 有 时 候 却 想 要 把 日 期 显示 为 一 个 字 从 串 ，date() 函 数 可 实现 
回 一 个 表示 日 期 的 、 格 式 化 的 字符 串 。 通 过 必须 传递 给 date0 的 字符 
mri RATE VASE EMT CI BEA AR PE Hl BR I BR EF 
串 ，date0 还 可 选 地 接受 一 个 时 间 惟 。 表 10-4 列 出 了 一 个 格式 化 字符 串 可 
能 包含 的 一 些 代 码 。 我 们 可 以 在 http:/www.php.netdate 找到 完整 的 列 
表 。 包 售 在 传递 给 date0 的 格式 字符 串 中 的 任何 其 他 数据 都 包含 在 了 返 
回信 中 。 


表 10-4 ”使 用 date0 的 某 些 格式 化 代码 


格 本 
ii HH 7m W 
人 ee 


一 月 中 的 第 几 天 《用 0 填充 空 日 的 数字 ) 





时 区 标识 符 America/Los_Angeles 
re 


h | 小 时 《12 小 时 格式 一 前 面 的 空白 用 0 填充 ) 














一 月 中 的 第 几 天 《用 0 填充 前 面 的 空白 的 数字 ) 


m | 一 年 中 的 月 份 〈 数 字 一 一 用 0 填充 前 面 的 空 日 ) 


qe 一 年 中 的 月 份 (3 个 字母 ) Feb 


Bly TA AY 2 


vf 


从 合 RFC 822 标 准 的 完整 日 期 Tue, 28 Feb 2006 








Chttp://www.faqs.org/rfcs/rfc822.html ) 06:45:26 -0800 





U | 时 间 惟 1141137926 





程序 清单 10.5 试 验 了 上 述 格 式 代 人 码 中 的 一 部 分 
程序 清单 10.5 ”使 用 dateO 格 式 化 一 个 日 期 


: <?php 

: $time = time(); //stores the exact timestamp to use in this script 
: echo date("m/d/y G:i:s e", $time); 

echo "<br/>"; 

echo "Today is "; 

= echo date("jS \of F Y, \a\\t g:ia \i\\n e", $time); 

?> 


NOOB OND — 


程序 清单 10.5 两 次 调用 date(): RERIT., mh Shae 
日 期 格式 ; 第 三 次 在 第 6 行 调用 ， 产 生 一 个 更 长 一 些 的 格式 。 把 这 个 程 
序 清 蛙 保 存 到 一 个 名 为 datetest.php 的 艾 本 文件 中 并 且 在 Web 浏 顺 右 中 打 
开 它 。 日 期 显然 会 和 下 面 的 有 所 不 同 ， 但 是 ， 这 里 有 一 些 示例 得 出 如 
Pe 


91/17/12 21:29:31 America/New_York 
Today is 17th of January 2012, at 9:29pm in America/New York 


尺 官 这 个 格式 化 字符 串 看 上 去 很 神秘 ， 但 它 很 容易 构建 。 如 来 想 要 
添加 一 个 字符 串 ， 其 中 包 舍 的 字母 也 下 格式 化 代码 ， 我 们 可 以 通过 在 其 
前 面 放置 一 个 矢 杠 来 转 义 它们 。 对 于 转 义 时 变 成 了 控制 字符 的 那些 字 
人 符 ， 必 须 转 义 它们 前 面 的 反 斜 杜 。 例 如 ，\t 是 一 个 制 表 符 的 格式 化 代 
人 码 ， 因 此 ， 为 了 确保 显示 出 制 表 和 从， 在 程序 清单 10.5 所 示 的 例子 中 使 用 
\\to 


另 一 个 例子 是 ， 在 一 些 环境 下 ， 一 个 单词 要 添加 到 一 个 字符 串 中 ， 
例如 the。 单 词 the 是 由 3 个 格式 化 代码 组 成 的 ， 因 此 必须 转 义 ， 示 例如 
Te 


<?php 

echo date('l \t\h\e jS'}; 
/fprints Tuesday the 3rd 
> 


还 需要 注意 ，date0 函 数 根 据 本 地 时 区 返回 信息 。 如 果 想 要 用 GMT 
格式 化 一 个 日 期 ， 我 们 使 用 gmdate0 函 数 ， 它 工作 的 方式 几乎 与 date0) 郴 
数 相 同 。 


10.4.4 ”使 用 mktime() 创 建 时 间 截 


我 们 已 经 获得 了 了 有关 当前 时 间 的 信息 ， 但 是 还 无 法 使 用 任意 的 日 
期 。mktime0 返 回 一 个 时 间 惟 ， 随 后 我 们 可 以 用 date0 或 getdateO 使 用 
它 。mktime() 按 照 如 下 的 顺序 接受 6 个 整 型 参数 。 


® 小 时 
。 分 人 钟 
T 


。 Aft 


。 日 期 
。 年 份 


程序 清单 10.6 使 用 mktime0 来 获取 一 个 时 间 戳 ， 然 后 ， 我 们 通过 
date0 函 数 来 使 用 这 个 时 间 截 。 


程序 清单 10.6 ”使 用 mktime0 创 建 一 个 时 间 榭 


: <?php 

: {/ make a timestamp for Jan 17 2012 at 9:34 pm 

: Sts = mktime(21; 34, ©; Ie 17; 2012); 

: echo date("m/d/y G:ii:s e", $ts); 

; echo *<br/>'; 

: echo "The date is " 

: echo date("jS \of F Y, \a\\t giia \i\\n e", $ts ); 
?> 


O N QORA ON = 


我 们 在 第 3 行 调 用 mktime0O 并 且 把 返回 的 时 间 惟 赋 给 了 $ts 变 量 。 然 
后 ， 在 第 4 行 和 第 7 行 分 别 使 用 dateO 函 数 ， 输 出 了 使 用 $ts 的 日 期 的 格式 
化 版 本 。 我 们 可 以 选择 省 略 掉 mktime() 的 一 些 参数 或 所 有 参数 ， 并 且 用 
和 当前 时 间 对 应 的 值 来 蔡 代 。mktime0 还 可 调整 那些 超出 相应 范围 的 
值 ， 因 此 ， 一 个 25 的 小 时 参数 转换 为 年 份 、 月 份 、 日 期 参数 指定 的 日 期 
之 后 的 一 天 的 1:00 am. 


把 上 述 代 码 保 存 到 名 为 mktimetest.php 的 文件 中 并 且 在 Web 浏 览 器 中 
打开 它 。 我 们 将 会 看 到 如 下 结果 。 


01/17/12 21:34:00 America/New York 
The date is 17th of January 2012, at 9:34pm in America/New York 


10.4.5 “使 用 checkdate0 测 试 日 期 


我 们 可 能 需要 从 用 户 得 入 接受 日 期 信息 。 在 使 用 用 户 和 输入 的 日 期 或 


者 将 其 存储 到 数据 库 之 前 ， 确 保 这 个 日 期 是 有 效 的 。checkdate() 接 受 3 个 
整 型 参数 月份、 日 期 和 年 份 。 如 末 月 份 在 1 到 12 之 间 ， 并 且 对 于 给 定 
的 月 份 和 年 份 (考虑 到 国 年 )， 这 个 日 期 是 可 以 接受 的 ， 以 及 年 份 在 0 
有 |32767 之 间 ，checkdate() 返 回 true。 注 意 ， 日 期 可 能 在 checkdate() 函 数 
中 是 有 效 的 ， 但 是 在 其 他 的 日 期 函数 中 和 是 不 可 接受 的 。 例 如 ， 如 下 的 代 
hH% [HI true. 


checkdate(4, 4, 1066) 


如 果 试 图 使 用 上 述 参 数值 通过 mktime() 来 构建 一 个 日 期 ， 最 终 将 得 
到 一 个 -1 的 时 间 惟 。 首 要 的 原则 是 不 要 对 mktime0 使 用 1902 年 以 前 的 年 
份 ， 并 且 ， 配 合 日 期 函数 使 用 1970 年 以 前 的 任何 日 期 的 时 候 都 要 小 心 ， 
因为 负数 不 是 有 效 的 日 期 。 由 于 UNIX 时 间 惟 是 从 1970 年 1 月 1 日 开始 
的 ， 此 前 的 任何 日 期 都 是 无 效 的 〈 负 的 ) 时 间 惟 。 


10.5 HWETTE. HEKA HTI ee 2 


PHPA2rIR = PAB, TORE FFIT EA AE fs HR 
AY. EERPHP FMD Rie, MAZA HEIR o 


e Strings, http://www.php.net/manual/en/ref.strings.php 
e Date/Time, http://www.php.net/manual/en/ref.datetime.php 


除了 留意 添加 到 PHP 中 的 函数 ， 用 户 为 菜 些 函 数据 写 的 注意 事项 ， 
往往 会 为 各 种 编程 任务 提供 解决 方案 ， 在 我 们 构建 目 己 的 应 用 程序 时 会 
发 现 这 些 很 有 用 。 


10.6 ”小结 


在 本 章 中 ， 我 们 已 经 7 了解 到 了 可 以 在 PHP 脚 本 中 控制 学 符 串 的 一 些 
函数 。 学 习 了 如 何 使 用 printf() 和 sprint0 格 式 化 字符 串 ， 我 们 应 该 能 够 使 
用 这 些 函数 来 创建 要 转换 数据 的 或 者 格式 化 数据 的 字符 串 。 我 们 学 习 了 
获取 字符 串 信息 的 函数 ， 可 以 使 用 strlen(0) 来 获取 一 个 字符 串 的 长 度 ， 使 
用 strpos0) 来 判断 一 个 子 字 人 符 串 的 存在 ， 或 者 使 用 substr() 来 提取 一 个 子 
字符 串 ， 还 可 以 使 用 strtok0 来 分 解 一 个 字符 串 。 


我 们 还 了 解 了 转换 字符 串 的 图 数 。 现 在 可 以 使 用 rim0、ltrim0 或 
rtrim() 从 字符 串 的 开头 或 结尾 删除 空 日 。 可 以 使 用 strtoupper()、 
strtolower0O 或 ucwords0 改 变 一 个 字符 串 中 的 字符 的 大 小 号。 可 以 用 
str_replace(0) 来 蔡 换 一 个 字符 串 中 的 所 有 实例 。 


我 们 还 学 习 了 如 何 使 用 各 种 PHP 函 数 来 执行 与 日 期 和 时 间 相 关 的 操 
作 。time0 函 数 从 当前 日 期 和 时 间 获 取 一 个 时 间 戳 ， 并 且 ， 我 们 可 以 使 
用 getdate0) 来 从 一 个 时 间 惟 提取 日 期 信息 ， 使 用 date0 把 一 个 时 间 惟 转换 
为 一 个 格式 化 字符 串 。 我 们 学 习 了 如 何 使 用 mktime0O 创 建 一 个 时 间 惟 ， 
以 及 如 何 使 用 checkdateO 验 证 日 期 的 有 效 性 。 我 们 还 将 在 第 16 章 和 学习 多 
个 功能 更 强大 的 、 和 日 期 相关 的 函数 ， 以 全 于 可 能 发 现 对 于 目 己 的 很 多 
与 时 间 日 期 相关 的 需求 可 以 使 用 MySQL 而 不 古 PHP。 


10.7 ”实践 练习 


这 个 实践 练习 设计 用 来 帮助 你 预测 可 能 的 问题 、 复 习 已 经 学 过 的 知 
识 ， 并 且 开 始 把 知识 用 于 实践 。 


10.8 Q&A 


Q: 我 们 可 以 组 合 多 个 字符 串 函 数 吗 ? 


A: FTV. FRAT REE eA, MAN Ae er eR. Bid 
FHE TP aR RE Ss AG RSS CAE Cae LE PH KE PR BL o 


Q: 对 于 保存 在 数据 库 中 而 不 是 脚本 中 的 日 期 和 时 间 ， 我 该 
如 何 使 用 呢 ? 


A: 在 本 书 第 16 章 ， 我 们 将 学 习 MySQL 日 期 和 时 间 函 数 ， 但 是 ， 通 
苗 的 一 种 好 办 法 是 ， 如 果 你 要 从 数据 库 中 提取 日 期 和 时 间 ， 并 且 想 要 执 
行 和 日 期 和 时 间 相 关 的 操作 例如， 计算 时 间 ， 或 者 将 时 间 从 一 种 类 型 
转换 为 妨 一 种 类 型 ) ， 可 以 让 数据 库 为 你 做 这 些 事情 。 但 是 ， 这 也 并 不 
妨碍 你 对 于 从 数据 库 提 取 的 日 期 来 使 用 PHP 的 日 期 和 时 间 函 数 。 


问答 题 
1. 为 了 使 用 printfO 把 一 个 整数 格式 化 为 一 个 双 精 上 度 小 数 ， 应 该 使 
用 什么 转换 指示 人 符 ? 给 出 转换 整数 33 所 需 的 完整 语法 。 


2. 如 何 使 用 0 质 充 问题 1 中 的 转换 ， 以 使 小 数 点 前 面 的 部 分 有 4 个 字 
符 的 长 度 ? 


3. 如何 为 上 一 个 问题 中 已 经 格 却 化 的 译 点 数 指 定 一 个 两 位 小 数 的 


精度 ? 
4. 从 一 个 字符 串 中 提取 一 个 子 字符 串 应 该 使 用 什么 函数 ? 
5. 如 何 从 一 个 字符 串 的 开 尖 删除 空 日 ? 
6. 如 何 把 一 个 市 有 分 阳 符 的 字符 串 分 解 成 一 个 子 字 符 串 的 数组 ? 


7. 使 用 PHP 时 我 们 怎样 获取 一 个 表示 当前 日 期 和 时 间 的 UNIX 时 间 
ER? 


8， 哪 个 PHP 函 数 接受 一 个 时 间 惟 ， 并 且 返 回 一 个 表示 给 定 日 期 的 
天 联 数 组 ? 


9. 使 用 哪个 PHP 孙 数 来 格式 化 日 期 信息 ? 


10. 使 用 哪个 PHP 抑 数 来 检查 一 个 日 期 的 有 效 性 ? 


oy 


ee 
1. 用 来 把 一 个 整数 格式 化 为 一 个 双 精度 数 的 转换 指示 符 是 f， 转 换 
整数 33 的 语法 如 下 。 


printf ("*f", 33); 


2， 我 们 可 以 使 用 填充 指 示 符 来 填充 printfO 输 出 ， 即 一 个 空格 或 一 
个 0 后 面 跟 看 一 个 表示 我 们 希望 填充 的 字符 数目 的 数字 。 


printf ("s@4Tf", 33}; 

3. PETRA AF A.J TEE PS Rs EE) hg E 
字 。 它 应 该 帮 在 转换 指示 和 从 的 前 面 。 
printf("%04.2f", 33); 

4. substrORKŽ H RE FE FE |B] ANTT. 

5. jltrim0O 国 数 从 一 个 字符 串 的 开头 删除 空白 。 

6. explode0 国 数 把 一 个 字符 串 分 解 成 一 个 数组 。 

7. 使 用 time0O 函 数 。 


8. getdate() 录 数 返回 一 个 天 联 数 组 ， 其 元 系 包 含 了 给 定 日 期 的 各 个 


部 分 。 


Ukr 


9. 使 用 date()。 


10. 使 用 checkdate() 函 数 来 检查 一 个 日 期 的 有 有 效 性 。 


EA wel 


1. 创建 一 个 接受 用 户 的 全 名 和 E-mail 地 址 的 反馈 表单 。 使 用 大 小 
See, APRN ASN AFAR, FHEAR ER E 
Mite. AEA PIN E-mailiitheaaa—-T@t Ss, WRN E N 
不 一 条 警告 。 

2. 创建 双 精 度数 和 整数 的 一 个 数组 。 通 历 这 个 数组 ， 把 每 个 元 系 
祁 转 换 为 具有 2 个 精度 的 一 个 译 点 数 ， 结 末 输 出 在 一 个 20 字 符 的 字段 
H, FP AAMT SF 


3. 创建 一 个 出 生日 倒计时 脚本 。 给 定 一 个 表单 输入 月 份 、 日 期 和 
年 份 ， 输 出 信息 告诉 用 户 距 离 出 生日 有 和 多少 天 、 多 少 小 时 、 多 少 分 和 多 


DED 


第 11 章 ”使 用 表单 


ERER, MÄ FI: 


如 何 从 表单 字段 获取 信息 。 

如 何 使 用 允许 多 个 选择 的 表单 元 系 。 

如 何 创建 一 个 单个 的 文档 ， 包 含 一 个 HIML 表 单 以 及 处 理 其 提交 
的 PHP 代 码 。 

如 何 傈 存 隐 藏 字段 的 状态 。 

如 何 把 用 户 重 定 同 到 一 个 新 的 页面 。 

如 何 构 建 发 送 邮 件 的 HTML 表 单 和 PHP 代 码 。 

如 何 构建 一 个 上 传 文件 的 HTML 表单 ， 并 且 编 写 处 理 上 传 的 代 
A, 


到 现在 为 止 ， 本 书 中 的 PHP 示 例 虱 缺少 一 个 关键 要 系 。 当 然 ， 我 们 
己 经 学 习 了 基础 知识 ， 可 以 设置 变星 和 数组 ， 创 建 并 调用 函数 ， 并 且 可 
以 使 用 字符 串 。 但 是 ， 如 琳 用 户 无 法 以 一 种 有 意义 的 方式 和 Web 站 氮 区 
互 的 话 ， 这 一 切 都 室 无 意义 。HTML 表 单 就 是 将 大 量 信 息 从 用 户 传递 给 
ARS as HY HE PE. Al, ERER, BOSE H tee xe BR, 
FF ALR rR AAEH FA EA SHG, 


11.1 创建 一 个 答 单 的 输入 表单 


从 现在 开始 ， 让 我 们 把 HIML 和 了 PHP 代码 分 开 。 程 序 清 单 11.1 构 建 


了 一 个 简单 的 HIML 表 单 。 


程序 清单 11.1 简单 的 HTML 表单 





ON HM OB GGN — 


<!DOCTYPE html> 
<html> 
<head> 


<title>A simple HTML form</title> 
</head> 


<body> 

<form method="post" action="send_simpleform.php'> 
<p><label for="user">Name:</label><br/> 

<input type="text" id="user" name="user'></p> 


: <p><label for="message">Message:</label><br/> 

: <textarea id="message" name="message" rows="5" cols="40"></textarea></p> 
: <button type="submit" name="Submit" value="send">Send Message</button> 

: </form> 

: </body> 

: </html> 


把 上 述 代 码 放 入 到 一 个 名 为 simpleform.html 的 文本 文件 中 ， 并 且 


将 其 放置 在 Web 服 务 冀 文档 根 目 录 下 。 这 个 代码 清早 定义 了 一 个 表单， 


me 


a 


包含 在 第 9 行 定义 的 一 个 名 为 “user” 的 文本 字段 ， 在 第 11 行 定义 的 一 个 


名 为 “message” 有 的 文本 域 ， 以 及 在 第 12 行 定义 的 一 个 提交 按钮 。FORM 元 
素 的 ACTION 参 数 指 同一 个 名 为 send_simpleform.php 的 文件 ， 它 会 处 理 
表单 信息 。 这 个 表单 的 方法 是 POST， 因 此 ， 表 单 中 的 数据 变量 存储 在 
超 全 局 变量 $_ POST 中 。 


程序 清单 11.2 创 建 了 接受 用 户 输入 的 代码 。 


程序 清单 11.2 ”从 一 个 表单 谈 取 和 输入 


<!DOCTYPE html> 

<html> 

<head> 

<title>A simple response</title> 

</head> 

<body> 

<p>Welcome, <strong><?php echo $ POST['user']; ?></strong>!</p> 

<p>Your message is: 

<strong><?php echo $ POST[ message ]; ?></strong></p> 
: </body> 
: </html> 


一 © ON OO hh ON — 


一 & 


把 上 述 代 人 码 放 入 到 一 个 名 为 send_simpleform.php 的 文本 文件 中 ， 并 
日 将 其 放置 在 Web 服 务 右 文档 根 目录 下 。 现 在 使 用 浏览 右 访 问 这 个 表单 
本 里 (simpleform.html)， 然 后 ， 我 们 会 看 到 如 图 11-1 所 示 的 结 





A simple response 


€ C | © http://localhost/11/send_simpleform.php a, 


Welcome, John Smith! 


Your message is: I like cheese! 


图 11-1  simpleform.html @!) @ Hy 4 A 


当 用 户 提 交 程 序 清 单 11.1 所 创建 的 表单 时 ， 将 会 调用 程序 清单 11.2 
中 的 脚本 。 在 程序 清单 11.2 的 代码 中 ， 我 们 访问 了 两 个 变量 : $_POST 


[user] 和 $_POST [message]。 这 些 都 是 对 超 全 局 变量 $_ POST 中 的 变量 
的 引用 ， 这 些 变 量 包 售 了 用 户 输入 到 user 文 本 字段 和 message 文 本 域 中 的 
值 。PHP 中 的 表单 束 是 如 此 之 徐 单 。 


FERRE BCP a ASR ea SF Se. BUI MA A N A ONY 
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提示 : 





我 们 也 可 以 在 这 个 表单 以 及 其 他 表单 中 使 用 GET 方 法 。POST 可 以 处 理 比 GET 更 多 的 数 
据 ， 并 且 不 会 把 数据 传递 到 查询 字符 串 中 。 如 果 你 使 用 了 GET 方 法 ， 确 保 将 超 全 局 变量 修改 
为 $_GET 而 不 是 $_ POST。 








11.2 ”使 用 用 户 定 义 数组 访问 表单 输入 


前 面 的 例子 展示 了 如 何 从 HTML 元 素 收 集 信息 ， 这 些 元 素 问 每 个 元 
系 名 提交 一 个 单独 的 人 ， 例 如 文本 字段 、 文 本 域 和 单 选 按钮 。 当 使 用 像 
SELECT 这 样 的 元 系 的 时 候 ， 这 吏 给 我 们 这 来 一 个 问题 ， 因 为 用 户 可 能 
从 一 个 多 选 SELECT 列表 中 选取 一 个 或 多 个 项 目 。 如 果 我 们 使 用 一 个 一 
般 的 名 字 来 命名 SELECT 元 素 ， 如 下 所 示 。 


<input type="checkbox" id="products" name="products"> 


接受 这 个 数据 的 脚本 只 是 获取 了 和 这 个 名 字 ($_POST ['products']) Xt 
应 的 一 个 单个 值 。 我 们 可 以 通过 重新 命名 这 类 元 素 ， 使 其 名 字 以 一 对 裕 
的 方 括号 结尾 ， 从 而 改变 这 种 行为 。 我 们 在 程序 清单 11.3 中 这 么 做 。 


程序 清单 11.3 包含 一 个 SELECT 元 素 的 HTML 表单 


<!DOCTYPE html> 

<html> 

<head> 

<title>An HTML form with checkboxes</title> 
</head> 

<body> 

<form action="send_formwithcb.php" method="POST > 
<p><label>Name:</label><br/> 

<input type="text" name="user" /></p> 

10: <fieldset> 

11: <legend>Select Some Products:</legend><br/> 
12: <input type="checkbox" id="tricorder” 


ON oath Wh 一 


< 


less name="products[]" value="Tricorder'"> 

14: <label for="tricorder">Tricorder</label><br/> 
Lo. 

16: <input type="checkbox" id="ORAC AI" 

| name="products[]" value="ORAC AI > 

18: <label for="ORAC AI">ORAC AI</label><br/> 

19: 

20: <input type="checkbox" id="HAL 2000" 

21: name="products[]" value="HAL 2000 > 

22: <label for="HAL 2000">HAL 2000</label> 


23: </fieldset> 

24: <button type="submit" name="Submit" value="Submit">Submit Form</button> 
25: </form> 

26: </body> 

27: </html> 


把 上 述 代码 放 入 到 一 个 名 为 formwithselect,html 的 文本 文件 中 ， 并 且 
将 其 放置 在 web 服务 硕 文 档 根 目录 下 。 接 下 来 ， 在 处 理 表 单 输入 的 脚本 
中 ， 我 们 有 发现 ， 名 为 "products [的 所 有 选中 的 复 选 框 的 什 ， 在 一 个 名 
为 $ POST [products] 的 数组 中 是 可 用 的 。 在 第 12 行 到 第 22 行 ， 指 定 了 每 
个 复 选 框 。 在 程序 清单 11.4 中 ， 可 用 的 用 户 选 择 在 一 个 数组 中 给 出 。 


程序 清单 11.4 从 程序 清单 11.3 的 表单 中 读 取 输入 


<!DOCTYPE html> 

<html> 

<head> 

<title>Reading checkboxes</title> 


ON 一 


5: </head> 

6: <body> 

7: <p>Welcome, <strong><?php echo $ POST['user']; ?></strong>!</p> 
8: <p>Your product choices are: 

9: <?php 

10: if (tempty($ POST['products'])) { 


| iz echo "<ul>"; 

12: foreach ($ POST['products'] as $value) { 
1a: echo "<li>$value</li>"; 

14: } 

15: echo "</ul>"; 

16: } else { 

TE: echo "None"; 

ies ， 

19: ?> 


20: </body> 
21: </html> 


把 上 述 代 码 放 入 到 一 个 名 为 end_formwithselect.php 的 文本 文件 中 ， 
并 且 将 其 放置 在 Web 服 务 需 文档 根 目 孙 下 。 现 在 ， 通 过 Web 浏 览 硕 访问 
这 个 表单 并 有 项 写字 段 。 图 11-2 给 出 一 个 例子 。 























f An HTML form with checkt & \ 


€e > C O hitp://localhost/11/formwithcb.htm! K A A 


Name: 


Select Some Products: 











W] Tricorder | 
E] ORAC AI | 
HAL 2000 
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| 
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图 11-2 ”程序 清单 11.3 所 创建 的 表单 


在 程序 清单 11.4 中 的 脚本 的 第 7 行 ， 我 们 访问 了 $_POST [user] 变 
量 ， 这 个 变量 源 自 user 表 单元 素 。 在 第 10 行 ， 我 们 测试 了 $_POST 
[products] 变 量 。 如 条 $_POST [products] 变 量 存在 ， 我 们 在 第 12 行 过 历 
它 ， 并 且 在 第 13 行 把 每 种 选择 输出 到 浏览 器 。 选 定 的 checkbox 元 素 的 
value 属 性 中 的 文本 成 为 了 存储 在 数组 中 的 值 。 


提交 表单 ， 结 果 如 图 11-3 所 示 。 


fa] Reading checkboxes k 
2 C |O http://localhost/11/send_formwithcb.php A 
Welcome. Jane User! 
| Your product choices are: 


è Tricorder 
* HAL 2000 


图 11-3 ”send_formwithselect.php 的 输出 示例 


人 循环 技术 不 仅 对 于 SELECT 元 系 特 别 有 用 ， 也 可 以 对 其 他 类 型 的 表 
单元 系 起 作用 。 例 如 ， 通 过 给 定 共 有 相同 名 字 的 多 个 复 选 枉 ， 我 们 使 得 
一 个 用 户 可 以 在 一 个 单独 的 字段 名 中 选择 多 个 信 。 


只 要 我 们 选择 的 名 字 是 以 一 个 空 日 方 括号 结束 的 ，PHP 痢 会 把 这 个 


字段 的 用 户 输入 编 详 到 一 个 数组 中 。 


11.3 ”在 单个 页 面 上 组 合 HTML 和 PHP 代码 


在 菜 些 条 件 下 ， 我 们 可 能 想 要 把 解析 表单 的 PHP 代 码 和 直接 编码 的 
HTML 表 单 包含 到 同一 个 页 面 中 。 如 果 需 要 对 多 个 用 户 显 示 同 一 个 表单 
的 话 ， 这 种 组 合 可 能 会 有 用 。 当 然 ， 如 果 动 态 地 编写 整个 页 面 ， 将 会 有 
更 多 的 灵活 性 ， 但 是 我 们 会 错过 PHP 最 强大 的 功能 之 一 ， 这 就 是 和 标准 
HTML 的 民 好 结合 。 可 以 在 页 面 中 包含 的 标准 HTML 越 多 ， 设 计 者 或 页 
面 开发 者 修复 起 来 束 越 轻松 ， 而 不 必 寻 找 程序 员 帮 忙 。 


对 于 下 面 的 例子 ， 假 设 我 们 要 创建 一 个 站 点， 它 同和 尝 龄 前 儿童 教授 
基本 的 数学 知 识 。 它 要 求 我们 创建 一 个 脚本 ， 该 脚本 从 表单 输入 一 个 数 
衬 ， 并 且 香 诉 用 户 它 比 一 个 预定 义 的 整数 大 还 是 小 。 


程序 清单 11.5 创 建 了 HTML 。 对 于 这 个 例子 ， 我 们 只 需要 一 个 文本 
字段 ， 但 是 即便 如 此 ， 我 们 还 是 包含 了 一 些 PHP。 


程序 清单 11.5 调用 自身 的 一 个 HTML 表 单 


n <!DOCTYPE html> 

: <html> 

: <head> 

: <title>An HTML form that calls itself</title> 

: </head> 

: <body> 

: <form action="<?php echo $ SERVER['PHP_SELF']; ?>" method="POST"> 
: <p><label for="guess">Type your guess here:</label> <br/> 

9: <input type="text" id="guess" name="guess" /></p> 

10: <button type="submit" name="Submit" value="submit">Submit</button> 
11: </form> 

12: </body> 

13: </html> 


ON oar Wh 一 


正如 第 7 行 所 示 ， 这 个 脚本 的 动作 是 $_ SERVER ['PHP_SELF'], ix 


个 全 局 变量 显示 了 当前 脚本 的 名 字 。 换 句 话 说 ， 这 个 操作 告诉 脚本 重新 
载 入 目 己 。 程 序 清 单 11.5 中 的 脚本 并 不 产生 任何 输出 ， 但 是 ， 如 来 问 
Web 服 务 费 上 传 了 这 个 脚本 ,访问 该 页 面 并 察看 页 面 的 源 代 人 码 ， 我 们 将 
会 注意 到 ， 表 单 动作 现在 包含 了 脚本 目 映 的 名 子 。 在 程序 清单 11.6 中 ， 
我 们 开始 构建 这 个 页 面 的 PHP 元 系 。 


程序 清单 11.6 一 个 PHP 猿 数字 脚本 


1: <?php 

2: $num to guess = 42; 

3: if (tisset($ POST['guess'])) { 

4: $message = "Welcome to the guessing machine!"; 

5: } elseif (!is_numeric($ POST['guess'])) { // is not numeric 

Gt $message = "I don't understand that response."; 

7: } elseif ($ POST['guess'] == $num_to guess) { // matches! 

8: $message = "Well done!"; 

9: } elseif ($ POST['guess'] > $num to guess) { 

10: $message = $_POST['guess']." is too big! Try a smaller number."; 
11: } elseif ($ POST['guess'] < $num to guess) { 

12: $message = $_POST['guess']." is too small! Try a larger number."; 
13: } else { // some other condition 

14: $message = "I am terribly confused."; 

toe g 

16: ?> 


首 匈 ， 必 须 定义 让 用 户 猜 的 数字 ， 我 们 在 第 2 行 做 了 这 件 事 情 ， 拒 
42 赋 给 了 $num_to_guess 变 量 。 接 下 来 ， 必 须 定 义 表 单 是 含 提 区 了 ， 我 
们 可 以 通过 查看 变量 $ POST ['guess] 的 存在 来 测试 提交 ， 只 有 在 脚本 已 
经 用 guess 字 上 段 的 一 个 值 提 有 交 的 时 候 ， 这 个 变量 才 可 用 。 如 果 $_POST 
['guess'] 的 值 不 存在 ， 我 们 可 以 安全 地 假设 访问 页 面 的 用 户 没 有 提交 一 
个 表单 。 如 采 这 个 值 存 在 ， 我 们 可 以 测试 该 变量 所 包含 的 值 。 对 
$_POST [guess'] 弯 量 存在 性 的 测试 在 第 3 行进 行 。 


第 3 行 到 第 15 行 使 用 了 一 个 if...else if...else 控 制 结 构 。 根 据 表单 提交 
了 什么 内 容 〈 如 果 有 提交 内 容 的 话 ) ， 在 任何 时 候 ， 这 些 条 件 中 只 有 一 


个 为 tue。 根 据 条 件 ， 不 同 的 值 会 赋 给 $message 变 量 。 随 后 ， 该 变量 在 
脚本 的 第 18 行 显示 到 屏幕 上 ， 那 是 这 段 脚 本 的 HIML 的 一 部 分 。 


Ii 
Lo 
19: 
20: 
2l: 
22: 
23: 
24: 
i e 
26: 
pal ee 
28: 
29: 
3: 


程序 清单 11.7 一 个 PHP 猜 数字 脚本 ( 续 ) 


<!DOCTYPE html> 

<html> 

<head> 

<title>A PHP number guessing script</title> 

</head> 

<body> 

<hi><?php echo $message; ?></h1> 

<form action="<?php echo $ SERVER['PHP_SELF']; ?>" method="POST"> 
<p><label for="guess">Type your guess here:</label><br/> 

<input type="text" i1s="guess" name="guess" /></p> 

<button type="submit" name="Submit" value="Submit">Submit</button> 
</form> 

</body> 

</html> 


把 程序 清单 11.6 和 11.7 中 的 所 有 代码 放 入 到 一 个 名 为 numguess.php 


的 文本 文件 中 ， 并 且 将 这 些 文件 放 入 到 Web 服 务 卓 文档 根 目录 下 。 现 
在， 使 用 浏 史 各 访 问 这 些 脚本 ， 将 看 到 如 图 11-4 所 示 的 结 
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图 11-4 程序 清单 11.6 所 生成 的 表单 


仍然 还 有 一 些 和 额外 的 事情 可 以 做 ， 但 是 ， 你 可 能 会 明白 ， 把 代码 区 
给 一 个 设计 师 让 他 进行 艺术 加 工 ， 这 是 一 件 多 么 简单 的 事情 。 设 计 师 可 
以 做 好 他 的 那 部 分 工作 ， 而 不 必 以 任何 方式 干扰 程序 设计 ，PHP 代 码 在 
最 上 面 ， 剩 下 的 99% 都 是 HTML。 


11.4 (75 bape BOR RAE TK AS 


通过 程序 清单 11.6 中 的 脚本 我 们 没有 办 法 知道 一 个 用 户 的 猜测 是 多 
少 ， 但 是 ， 我 们 可 以 使 用 一 个 隐 蔚 字段 来 记录 这 个 什 。 隐 闫 字段 的 行为 
和 一 个 文本 字段 一 样 ， 只 不 过 用 户 无 法 看 到 它 ， 除 非 租 看 包 合 它 的 文档 
的 HTML 源 代码 。 


取出 最 初 的 numguess.php 脚 本 并 将 其 保存 为 一 个 名 为 humguess2.php 
的 副本 。 在 新 的 版 本 中 ， 在 最 初 给 $num_to_guess 变 量 赋值 的 后 面 深 加 
如 下 一 行 。 
$num tries = (isset{$ POST['num tries']}} ? $num tries + 1: 1; 

这 一 行 初 始 化 一 个 名 为 $num_tries 的 变量 并 且 为 它 赋 一 个 值 。 如 果 
$_POST [num_tries] 为 空 且 这 个 表单 还 设 有 捉 交 ，$num_tries 变 量 的 值 
为 1， 因 为 我 们 将 要 开始 第 一 次 猜 数 。 如 果 这 个 表单 已 经 发 送 ， 新 的 值 
His POST [mum tries] 的 值 加 1。 

下 一 个 改变 在 HTML 的 H1 层 标题 之 后 。 


<p><straong>Guess number:</strong> <?php echo $num tries; T> p> 
这 个 新 行 只 是 在 屏幕 上 显示 $num tries 的 当前 值 。 


最 后 ， 在 表单 提交 按钮 的 HTML 代 码 之 前 ， 添 加 隐藏 字段 。 这 个 字 
段 保存 了 $num_tries 圳 增 后 的 值 ， 如 下 所 示 。 


<input type="hidden" name="num tries" value="<?php echo $num tries; ?>"/> 


程序 清单 11.8 显 示 了 完整 的 新 脚本 。 


程序 清单 11.8 MEHA Bei BAP TAS 


1: <?php 

2: $num to guess = 42; 

3: $num tries = (isset($ POST['num_tries'])) ? $num_tries + 1: 1; 

4: if (tisset{$ POST[ 'guess'])) { 

Di $message = "Welcome to the guessing machine!" ; 

6: } elseif (!is numeric($ POST['guess'])) { // is not numeric 

ys $message = "I don't understand that response. "; 

8: } elseif ($ POST['guess'] == $num to guess) { // matches! 

g $message = "Well done!"; 

10: } elseif ($ POST['guess'] > $num_to_guess} { 

i $message = $ POST['guess']." is too big! Try a smaller number."; 
12: } elseif ($ POST[ ' guess ] < $num to guess} { 

13% $message = $ POST['guess']." is too small! Try a larger number."; 
14: } else { // Some other condition 

ia $message = "I am terribly confused.”; 

16: } 

IES T> 


18: <!DOCTYPE html> 


19: <html> 

20: <head> 

21: <title>A PHP number guessing script</title> 
22: </head> 

23: <body> 


24: <hi><?php echo $message; ?></h1> 

25: <p><strong>Guess number:</strong> <?php echo $num_tries; ?></p> 

26: <form action="<?php echo $ SERVER['PHP SELF']; ?>" method="POST"> 

27: <p><label for="guess">Type your guess here:</label><br/> 

28: <input type="text" id="guess" name="guess" /></p> 

29: <input type="hidden" name="num_ tries" value="<?php echo $num tries; ?>"/> 
30: <button type="submit" name="Submit" value="Submit">Submit</button> 

31: </form> 

32: </body> 

33: </html> 


将 上 述 代码 保存 为 numguess2.php 文 件 并 且 将 其 放置 到 Web 服 务 需 文 
档 的 根 目 录 下 。 使 用 Web 浏 贤 器 访问 表单 几 次 ， 并 且 壬 试 猪 这 个 数字 
(假装 你 还 并 不 知道 它 ) 。 每 次 访问 表单 ， 计 数 右 将 会 增加 1。 


11.5 重 定 向 用 户 


我 们 的 简单 的 脚本 仍然 有 一 个 缺点 ， 不 各 用 户 狂 负 的 是 否 正 硝 ， 表 
单 都 重 新 载 入 。HTML 征 直接 编码 的 ， 这 一 事实 使 得 很 难 避 免 编 与 整个 
页 面 。 然 而 ， 我 们 可 以 把 用 户 重 定 同 到 一 个 视 贾 页面， 从 而 避免 这 个 问 


je 


当 一 个 服务 器 脚本 和 一 个 客户 机 通讯 的 时 候 ， 它 必须 首先 发 送 某 些 
标 头 ， 这 些 标 头 提供 了 有 关 后 续 文 档 的 信息 。PHP 通 第 会 为 你 自动 处 理 
这 些 标 头 ， 但 是 ， 可 以 选择 使 用 PHP 的 header0 函 数 发 送 自己 的 标 头 行 。 


为 了 调用 header0 函 数 ， 我 们 必须 绝对 确保 没有 辐 浏 览 喜 发送 输 
Ho BRAN RARAN RHR, PHPH A CHIER, KEA 
更 多 内 容 部 必须 在 此 之 后 。 任 何 来 自 文 档 的 输出 ， 即 便 是 脚本 标记 外 的 
一 个 换行 或 者 空格 ， 都 会 引起 标 头 发 送 。 如 果 你 想 要 在 脚本 中 使 用 
header() 浮 数 ， 必 须 确保 包含 冰 数 调用 的 PHP 代 码 之 前 没有 内 容 。 我 们 还 
应 该 检 栓 可 能 使 用 的 任何 其 他 库 。 


程序 清单 11.9 展 示 PHP 发 送 给 浏览 器 的 一 个 典型 的 标 头 ， 从 第 3 行 开 
始 ， 作 为 对 第 1 行 中 的 请 求 的 响应 。 


程序 清单 11.9 ”从 一 个 PHP 脚 本 及 送 的 典型 的 服务 硕 标 头 


: HTTP/1.1 200 OK 
; Date: Sun, 29 Jan 2012 15:50:28 PST 
: Server: Apache/2.2.21 (Win32) PHP/5.4.9 
: X-Powered-By: PHP/5.4.0 
: Connection: close 
: Content-Type: text/html 


DO 上 com 一 


通过 发 送 一 个 Location 标 头 而 不 是 PHP 的 默认 标 头 ， 我 们 可 以 使 浏 
览 蓓 重 定 同 到 一 个 新 的 页 面 ， 示 例如 下 。 
header("Location: http: /www,.samSspublishing.com' }; 

假设 我 们 已 经 创建 了 一 个 名 为 congrats.html 的 祝贺 页 面 ， 我 们 可 以 
修改 猜 数 字 脚 本， 使 得 如 末 猜 对 了 束 重 定 同 用 户 的 请 求 ， 如 程序 清单 
11.10 所 示 。 和 程序 清单 11.8 相 比较 ， 这 个 脚本 只 是 在 第 8 行 else 子 句 之 后 
Es 

程序 清单 11.10 4 H header0 Æ x€ [A] H 7 


1: <?php 

2: $num to guess = 42; 

3: $num tries = (isset($ POST['num_tries'])) ? $num_tries + 1: 1; 
4: if (!isset($ POST['guess'])) { 

B $message = "Welcome to the guessing machine!"; 

6: } elseif (!is numeric($ POST['guess'])) { // is not numeric 

T. $message = "I don't understand that response."; 

8: } elseif ($ POST['guess'] == $num to guess) { // matches! 

9; header("Location: congrats.html1") ; 

10: exit; 

11: } elseif ($ POST['guess'] > $num to guess} { 

12: $message = $ POST['guess']." is too big! Try a smaller number."; 
13: } elseif ($_POST['guess'] < $num to guess) { 

14: $message = $ POST['guess']." is too small! Try a larger number."; 
15: } else { // some other condition 

16: $message = "I am terribly confused."; 

17: } 

18: ?> 

19: 

20: <!DOCTYPE html> 

21: <html> 

22: <head> 

23: <title>A PHP number guessing script</title> 

24: </head> 

25: <body> 


26: <h1><?php echo $message; ?></h1> 

27: <p><strong>Guess number:</strong> <?php echo $num_tries; ?></p> 

28: <form action="<?php echo $ SERVER['PHP SELF']; ?>" method="POST"> 

29: <p><label for="guess">Type your guess here:</label><br/> 

30: <input type="text" id="guess" name="guess" /></p> 

31: <input type="hidden" name="num tries" value="<?php echo $num _tries; ?>"/> 
32: <button type="submit" name="Submit" value="sSubmit">Submit</button> 

33: </form> 

34: </body> 

35: </html> 


在 第 8 行 ， 我 们 的 让 语句 的 else 子 句 使 浏览 器 把 我 们 送 到 名 为 
congrats.html 的 负面。 我 们 确保 当前 页 面 的 所 有 输出 都 用 第 10 行 的 exit 语 
名 取消， 它 会 立即 结束 这 个 脚本 的 执行 和 输出 。 


11.6 IREKE fe AIA HF 
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之 再， 阅读 下 面 的 小 市 ， 确 你 已 正确 地 配置 了 我 们 的 系统 。 


11.6.1 mail) MAH) RA E. 


ERIA LEH mail( eR ACACIA ABE ZA, AE php.ini sc FF i 
置 一 些 指令 ， 以 使 该 函数 能 够 正常 地 工作 。 使 用 一 个 文本 编辑 器 打开 
php.ini， 并 和 奉 找 下 面 这 些 行 。 


[mail function] 

; For Wins2 only. 

; http: //php.net/smtp 

SMTP = localhost 

; http: //ohp.net/smtp-port 
smtp_port = 25 


; For Win32 only. 

; http: //php.net/sendmail-from 
;sendmail from = me@example.com 
; For Unix only. You may Supply arguments as well (default: "sendmail -t -1i"). 


; http: //php.net/sendmail-path 
;sendmail path = 


如 果 我 们 使 用 Windows 作 为 Web 服 务 絮 平台 ， 前 两 条 指令 适用 于 我 
i]. Cee 数 能 够 发 送 邮 件 ， 它 必须 可 — os BUN ACIS 
邮件 服务 器 。 如 果 你 计划 使 用 ISP 的 发 送 邮 件 服务 器 (在 下 面 的 例子 
H, mjas EarthLink) ，php.ini 中 的 条 目 应 该 如 下 所 示 。 


SMTP = smtp.yourisp.net 


第 二 条 配置 指令 是 sendmail from， 这 是 在 发 送 邮件 的 From 标 头 中 
的 E-mail 地 址 。 它 可 以 在 邮件 脚本 本 吴 中 被 禾 闸 ， 这 里 通 第 作为 默认 值 
使 用 ， 如 下 面 的 例子 所 示 。 


sendmail from = youraddress@yourdomain.com 


对 于 Windows 用 户 来 说 ， 一 个 较 好 的 首要 原则 是 ， 不 管 你 的 Email 
客户 问 在 机 器 上 安装 了 什么 发 送 邮 件 服 务 器 ， 都 应 该 作为 php.ini 中 的 
SMTP 值 使 用 。 


如 果 Web 服 务 器 运行 在 一 个 Linux/UNIX 平 台 上 ， 可 以 使 用 特定 机 器 
的 sendmail 功 能 。 在 这 个 例子 中 ， 只 有 最 后 的 命令 适合 于 你 ， 即 
sendmail_path. SA hB sendmail -t-i， 但 古 ， 如 果 sendmail 古 临时 的 
或 者 如 果 你 需要 指定 不 同 的 参数 ， 也 可 以 不 这 么 做 ， 束 保 在 下 和 面 的 例子 
中 ， 它 没有 使 用 真实 仁 。 


sendmail path = /opt/sendmall -odd -arguments 


在 任何 平台 上 ， 对 php.ini 做 出 改变 以 后 ， 必 须 重 新 司 动 Web 服 务 需 
以 使 改变 生效 。 


11.6.2 ”创建 表单 


在 程序 清单 11.11 中 ， 我 们 看 到 用 来 创建 一 个 简单 反馈 表单 的 基本 
HTML， 让 我 们 称 其 为 feedback.htm1。 这 个 表单 具有 一 个 sendmail.php 的 
action， 我 们 将 在 下 一 节 中 创建 它 。feedback.html 中 的 字段 很 简单 ， 第 8 
行 和 第 9 行 创建 了 一 个 name 字 段 和 标 丛 ， 第 10 行 和 第 11 行 创建 了 回信 的 
Email 地 址 字段 和 标签 ， 而 第 12 行 和 第 13 行 包含 了 用 于 用 户 消息 的 文本 
域 和 标签 。 


O N QOBON — 


ae) 


程序 清单 11.11 创建 一 个 简单 的 反馈 表单 


<!DOCTYPE html> 

<html> 

<head> 

<title>E-Mail Form</title> 

</head> 

<body> 

<form action="sendmail.php" method="PQST"> 

<p><label for="name">Name:</label><br/> 

<input type="text" size="25" id="name" name="name" /></p> 


: <p><label for="email">E-Mail Address:</label><br/> 

: <input type="text" size="25" id="email" name="email"/></p> 

: <p><label for="msg">Message:</label><br/> 

: <textarea id="msg" name="msg" cols="30" rows="5'></textarea></p> 

: <button type="Submit" name="submit" value="send'>Send Message</button> 
: </form> 

: </body> 

: </html> 


把 上 述 代 码 放 入 到 一 个 名 为 feedback.html 的 文本 文件 中 ， 并 且 将 其 


放置 在 Web 服 务 器 文档 根 目录 下 。 当 通过 Web 浏 览 器 来 访问 这 个 脚本 的 
时 候 ， 它 产生 如 图 11-5 所 示 的 结果 。 
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图 11-5 ”程序 清单 11.11 创 建 的 表单 
在 下 一 节 中 ， 我 们 将 创建 把 这 个 表单 发 送 给 接受 者 的 脚本 。 
11.6.3 ”创建 发 运 邮 件 的 脚本 


这 个 脚本 和 程序 清单 11.4 中 的 脚本 在 概念 上 只 是 略 有 人 不同， 后 者 只 
旦 在 屏 恬 上 显示 表单 啊 应 。 在 程序 清单 11.12 的 脚本 中 ， 除 了 把 啊 应 电 
示 到 屏 大 上 ， 我 们 还 将 其 友 壕 给 一 个 E-mail 地 址 。 


程序 清单 11.12 KX Ta) HLA Be tit ee HA 


1: <?php 

2: jistart building the mail string 

3: $msg = "Name: ".$ POST[ ' name ]e"\n"; 

4: $msg .= "E-Mail: ".$ POST[ 'email']."\n'; 

5: $msg .= "Message: ".$ POST[ 'message']."\n"; 
6: 

7: ffset up the mail 

8: $recipient = "“you@yourdomain.com"; 

9: $subject = "Form Submission Results"; 

1@: $mailheaders = “From: My Web Site <defaultaddress@yourdomain.com> \n'; 
11: $mailheaders .= "Reply-To: ".$ POST['email']; 
hes 


13: //send the mail 

14: mail{$recipient, $subject, $msg, S$mailheaders); 

Lot! Te 

16: <!DOCTYPE html> 

17: <html> 

18: <head> 

19: <title>Sending mail from the form in Listing 11.10</title> 
20: </head> 


21: <body> 
22: <p>Thanks, <strong><?php echo $_POST['name']; ?></strong>, 
23: for your message.</p> 


24: <p>Your e-mail address: 

25: <strong><?php echo $ POST['email']; ?></strong></p> 

26: <p>Your message: <br/> <?php echo $ POST[ 'message']; ?> </p> 
27: </body> 

28: </html> 


在 第 22 行 到 第 26 行 所 使 用 的 变量 是 $_POST ['name'], $_POST 
[email] 和 $_POST [message]， 它 们 是 表单 中 字段 的 名 字 ， 它 们 的 值 作 
为 超 全 局 变量 $_POST 的 一 部 分 保存 。 这 对 于 把 信息 显示 到 屏幕 上 来 说 
孝 是 很 不 错 的 ， 但 是 在 这 个 脚本 中 ， 我 们 还 需要 创建 一 个 在 Email 中 人 肥 
运 的 字符 串 。 为 此 ， 我 们 基本 上 需要 退 过 连接 字符 串 来 形成 一 条 长 长 的 
消息 字符 串 从 而 构成 邮件 ， 并 且 和 在 适当 的 地 方 使 用 换行 符 (\)。 


第 3 行 到 第 5 行 创 建 了 $msg 变 量 ， 这 是 包含 用 户 在 表单 字段 中 输入 的 
值 以 及 一 些 额外 的 次 明文 字 的 一 个 字符 昌 。 这 个 字符 串 将 会 形成 E-mail 
的 主体 。 注 意 在 第 4 行 和 第 5 行 向 $msg 变 量 添加 内 容 时 连接 操作 符 (.=) 的 
使 用 。 


第 8 行 和 第 9 行 对 邮件 接受 者 和 邮件 消 晨 的 主题 的 变量 直接 赋值 。 显 
然 ， 要 用 日 己 的 邮件 地 址 来 蔡 代 you@yourdomain.com。 如 果 想 要 改变 
邮件 的 主题 ， 也 可 以 直接 去 做 。 


第 10 行 和 第 11 行 设置 了 一 些 邮件 标 头 ， 即 From: 和 Reply-to: 标 头 。 
你 可 以 把 任何 值 放 入 到 From: 标 头 中 ， 这 吏 是 当 你 接 党 这 封 邮件 的 时 候 
显示 在 发 件 人 或 收 件 人 栏 的 信息 。 


如 果 发 送 邮 件 服务 器 是 一 台 Windows 机 器 ， 那 么 换行 符 \n 应 该 蔡 换 为 \rin。 


mail() 函 数 需 要 5 个 参数 : WRAL Fer. YAS. FEA RAE BB AP 
头 ， 以 及 任何 附加 的 发 送 邮件 参数 。 在 我 们 的 例子 中 ， 我 们 只 使 用 前 4 
个 参数 。 这 些 参数 的 顺序 如 第 14 行 所 示 。 


把 上 述 代 码 放 入 到 一 个 名 为 sendmail.php 的 文本 文件 中 ， 并 且 将 其 
放置 在 Web 服 务 问 文档 根 目录 下 。 使 用 Web 浏 吃 问 来 访问 这 个 表单 ， 并 
且 输 入 一 些 信息 ， 然 后 单 击 提交 按钮 。 我 们 将 会 在 浏览 器 中 看 到 如 图 
11-6 所 示 的 结果 。 


€ C |© localhost/11/sendmail.php ~ | 





Thanks, Jane Doe, for your message. 
| Your e-mail address: jane(a)doe.com 


Your message: 
I think your site is great! I will visit again soon. Also, can I have a pony? 


图 11-6 来自 sendmailphp 的 示例 结 


如 末 我 们 但 看 自己 的 E-mail， 应 该 会 有 一 条 消 居 在 等 看 我 们 去 疗 
读 。 它 看 上 去 如 图 11-7 所 示 。 


Sc 
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Form Submission Results 


| Reply-To: jane@doe.com 
| To: jemelon@gmail.com 


Name: Jane Doe 


E-Mail: jane@doe. com 


Message: | think your site is great! | will visit again soon. 


Also, can | have a pony? 


图 11-7” ”sendmail.php 发 送 的 邮件 


SPAN BIFFY Ee EAT ARH a ig AY ZEA RUE, FP ALR ze thei A 
户 已 经 在 表单 中 得 入 了 值 。 在 现实 的 情况 中 ， 在 对 邮件 做 任何 事情 之 
前 ， 你 可 能 要 检查 表单 中 是 否 输入 了 值 以 及 值 的 有 效 性 ， 这 需要 通过 
JavaScript 或 HTML5 表 单 验证 。 


11.6.4 ”使 用 HTMIL 格 式 化 邮件 


发 送 HTML 格 式 的 邮件 的 技巧 根本 就 算 不 上 技巧 。 实 际 上 ， 它 只 古 
涉及 编写 实际 的 HTML 和 修改 mail(0) 函 数 所 友 运 的 标 涉 。 作 为 程序 清 蛙 
11.12 的 一 个 改版 ， 程 序 清单 11.13 对 其 第 12 到 14 行 和 第 18 到 19 行 做 出 了 
修改 。 


程序 清单 11.13 AIA tal HAY Be tte HTML HA 


1: <?php 

2: jistart building the mail string 

3: $msg = "<p><strong>Name:</strong> “= POST | Name“ | srs pan 

4: $msg .= "“<p><strong>E-Mail:</strong> ".$ POST['email']."</p>"; 

5: $msg .= "<p><strong>Message:</strong> ".$ POST['message']."</p>"; 
6: 

7: ffset up the mail 

8: $recipient = "“you@yourdomain.com"; 

9: $subject = "Form Submission Results"; 

1@: $mailheaders = "MIME-Version: 1.@\r\n"; 

11: $mailheaders .= "Content-type: text/html; charset=I1S0-8859-1\r\n"; 
12: $mailheaders = "From: My Web Site <defaultaddress@yourdomain.com> \n"; 
13: $mailheaders .= "Reply-To: ".$ POST['email']; 

14: 


15: //send the mail 
16: mail{$recipient, $subject, $msg, S$mailheaders) ; 


17: ?> 

18: <!DOCTYPE html> 
19: <html> 

20: <head> 


21: <title>Sending the Simple Feedback Form - HTML Version</title> 
22: </head> 


23: <body> 
24: <p>Thanks, <strong><?php echo $_POST[ name ]; ?></strong>, 
25: for your message.</p> 


26: <p>Your e-mail address: 

27: <strong><?php echo $ POST['email']; ?></strong></p> 

28: <p>Your message: <br/> <?php echo $ _ POST[ message ]; ?> </p> 
29: </body> 

30: </html> 


ERTIES, HARSHER HTML. MIRER 
在 第 10 行 到 第 11 行 创建 ， 它 将 MIME Version 标 头 设置 为 1.0 并 将 Content- 
type 标 头 设置 为 字符 集 ISO-8859-1 的 texyhtml。 当 在 一 个 支持 HTML 的 邮 
件 和 客户 关上 打开 它 的 时 候 ， 消 息 字 人 符 串 中 的 HIML 将 如 期 地 巡 示 ， 如 和 图 
11-8 所 示 。 
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Form Submission Results 
1 message 


My Web Site <defaultaddressi@yourdomain.com=> 
Reply-To: janedoe. com 
| To: jemelon@gmail.com 


Sun, Jan 29, 2012 at 3:09 PM 


Name: Jane Doe 


E-Mail: jane@doe.com 


Message: Wow, HTML output in an email? So cool! Now | want a unicorn. 





图 11-8 程序 清单 11.13 发 送 的 E-mail 


11.7 使 用 文件 上 传 


到 目前 为 止 ， 我 们 已 经 看 到 了 一 些 简单 的 表 蛙 输出。 然而 ，Web 浏 
响 复 还 文 持 文件 上 传 等 ，PHP 当 然 也 文 持 。 在 本 世 中 ， 我 们 将 学 习 PHP 
用 来 处 理 这 种 输入 的 功能 。 


天 于 上 传 文件 的 信息 在 $_FILES 超 全 局 变量 中 ， 这 些 信息 通过 表单 
中 一 个 或 多 个 上 传 字 段 的 名 字 来 索引 。 这 些 键 中 的 每 一 个 相应 值 束 古 一 
个 天 联 数 组 。 表 11-1 插 述 了 这 些 字 段 ， 使 用 fleupload 作 为 用 于 上 传 的 表 
单字 段 的 名 字 。 


表 11-1 ”文件 上 传 全 局 变量 


$_FILES[“fileupload”][“name”] | 上 传 文件 的 最 初 名 字 


$_FILES[“fileupload”] 


[“tmp_name”] 








临时 文件 的 路 径 /tmp/phprDfZvN 


$ FILES[“fileupload”][“size”] | 上传 文件 的 大 小 〈( 字 节 ) 


上 传 文件 的 MIME 类 型 (由 客户 端 
给 定 ) 


$ FILES[“fileupload” ][“type”’] image/gif 





暂时 把 这 些 元 系 抛 到 脑 后 ， 我 们 将 在 下 一 市 创建 文件 上 传 表 蛙 。 


11.7.1 创建 文件 上 传 表单 


首先 ， 我 们 必须 创建 处 理 上 传 的 HTML 表 单 。 包 含 文件 上 传 字 段 的 
HTML 表 单 必 须 包 侣 如 下 一 个 ENCTYPE 参 数 。 


enctype="multipart/form-data" 


PHP 也 使 用 一 个 可 选 的 隐藏 字段 ， 这 个 字段 可 以 插入 到 文件 上 传 字 
段 之 前 。 这 个 字段 必须 叫做 MAX_FILE_SIZE， 并 且 应 该 有 一 个 值 ， 表 
示 你 愿意 接收 的 文件 的 最 大 字 节 数 。MAX_FILE_SIZE 字段 遵循 浏览 器 
的 设 定 ， 所 以 我 们 应 该 依赖 php.ini 设 定 的 upload_max_filesize， 以 管控 
不 合理 的 大 量 上 传 。 在 输入 MAX_FILE_SIZE 字 段 以 后 ， 我 们 添加 了 上 
传 字 段 本 身 。 这 只 是 一 个 TYPE 参 数 为 "file" 的 INPUT 元 素 ， 我 们 可 以 根 
据 目 己 的 意愿 给 它 起 一 个 名 字 。 程 序 清单 11.14 把 所 有 这 些 都 写 入 到 一 
个 HTML 上 传 表单 中 。 


程序 清单 11.14 一 个 简单 的 文件 上 传 表 单 


te SIDOCTYPE Rimi 

2: <html> 

3: <head> 

4: <title>A simple file upload form</title> 

5: </head> 

6: <body> 

7: <form action="do upload.php" enctype="multipart/form-data" method="POST"> 
8: <input type="hidden" name="MAX_ FILE SIZE" value="1048576" /> 


9: <p><label for="fileupload">File to Upload:</label> 

10: <input type="file" id="fileupload" name="fileupload" /></p> 

11: <button type="submit" name="Submit" value="send">Upload File</button> 
12: </form> 

13: </body> 

14: </html> 


正如 你 所 看 到 的 ， 在 第 8 行 ， 上 传 文件 大 小 限制 为 1MB。 并 且 上 
传 文件 字段 的 名 字 是 人 eupload， 如 第 10 行 所 示 。 把 这 个 程序 清单 保存 到 


— pZ Afileupload.htmlHy) 30 AS SCPE, FE ARR Web HS ir L 
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下 
j [4] A simple file upload form * 


File to Upload: | Choose File | No file chosen 
Upload File 


图 11-9 程序 清单 11.14 所 创建 的 表单 
这 个 表单 调用 了 do_upload.php 脚 本 ， 我 们 随后 创建 这 个 脚本 。 
11.7.2 ”创建 一 个 文件 上 传 脚 本 


如 果 还 记得 有 关 超 全 局 变量 $_FILES 的 信息 ， 你 就 具备 了 编写 一 个 
简单 的 文件 上 传 脚 本 所 需 的 所 有 信息 。 这 个 脚本 惑 是 程序 清单 11.15 所 
创建 的 表单 的 文 持 代 但 。 


程序 清单 11.15 一 个 文件 上 传 脚本 


1: <?php 

2: $file dir = "/path/to/upload/directory’ ; 

Ho 

4: foreach($ FILES as $file name => $file array) { 

5: echo "path: ".$file_array['tmp_name']."<br/>\n"; 
6: echo ‘name: ".$file_ array[ name ]. <br/>\n',; 

T: echo "type: ".$file array['type']."<br/>\n"; 

8: echo "size: ".$file_array['size']."<br/>\n"; 

si 

10: if (is_uploaded file($file array[ ' tmp name'])) { 
1 move uploaded file($file array['tmp name'], 
ts "$file dir/".$file_array['name']) 

iaz or die ("Couldn't move file"); 

14: echo "File was moved!"; 

Tat } else { 

16: echo "No file found."; 

1 1 } 

18: } 

19: ?> 


在 程序 清单 11.15 中 ， 我 们 首先 在 第 2 行 创 建 了 $file_dir 变 量 来 存储 路 
径 信息 。 这 条 路 径 应 该 是 你 的 系统 上 存在 的 一 条 路 径 ， 并 且 Web 服 务 右 
用 户 〈 例 如 ，httpd、www 和 nobody) 必须 拥有 对 它 的 写 权 限 。 


提示 : 





第 2 行 所 使 用 的 路 径 是 一 个 LinuxwUNIX 路 径 。Windows 用 户 应 该 使 用 转 义 的 反 斜 枉 ， 示 例 
如 下 。 


$file_dir = “C:\Users\Y ou\Desktop”; 


第 4 行 开 始 一 条 foreach 语 句 ， 它 授 历 了 $_FILES 数 组 中 的 每 个 元 又 。 
这 里 使 用 了 一 个 循环 而 不 是 一 条 让 语句 ， 来 使 我 们 的 脚本 可 以 扩展 到 在 
同一 个 页 面 处理 多 个 文件 上 传 。 第 4 行 的 foreach 循 环 把 上 传 文件 名 存储 
到 $file_name 变 量 中 ， 并 且 把 文件 信息 存储 到 $file_array 变 量 中 。 然 后 ， 
我 们 就 可 以 输出 所 拥有 的 关于 上 传 的 文件 信息 。 


在 把 上 传 文件 从 其 临时 位 置 移 动 到 第 2 行 指 定 的 位 置 之 前 ， 首 移 检 
谷 文 件 是 否 存 在 。 我 们 在 第 10 行 使 用 i_uploaded_fileO 函 数 来 做 这 件 事 
情 。 这 个 函数 接收 一 条 指 问 上 传 文件 的 路 径 ， 并 且 只 有 在 所 询问 的 文件 
是 一 个 有 效 的 上 传 文件 的 时 候 才 返回 true。 因 此 ， 这 个 函数 增强 了 脚本 
的 安全 性 。 


假设 这 些 部 正常 ， 在 第 11 行 到 第 13 行 ， 文 件 将 从 其 临时 位 置 复制 到 
一 个 狐 的 日 录 ， 我 们 使 用 为 一 个 函数 move_uploaded_file() 来 做 到 这 一 
扩 。 这 个 函数 把 一 个 文件 从 一 个 位 置 复制 到 男 一 个 位 置 ， 它 站 先 执 行 和 
is_uploaded_file() 函 数 所 做 的 相同 的 安全 检查 。move_uploaded _file() 杨 
数 需要 一 个 源 文 件 路 径 以 及 一 个 目标 路 径 。 如 末 移 动 成 功 ， 它 返回 
true， 如 末 文 件 不 是 有 效 的 上 传 文件 或 者 文件 没有 找到 ， 它 返回 false。 


注意 上 传 文件 的 名 字 。 像 Mac OS 和 Windows 这 样 的 操作 系统 ， 它 们 
对 行文 件 命名 问题 相当 宽松 ， 因 此 ， 上 传 文件 可 能 最 终 包 括 空格 、 引 筷 
以 及 各 种 各 样 其 他 的 不 可 预期 的 字符 。 因 此 ， 过 涯 文件 名 是 个 不 错 的 想 
法 。 


把 上 述 代码 放 入 到 一 个 名 为 do_upload.php 的 文本 文件 中 ， 并 且 将 其 
放置 在 Web 服 务 需 文档 根 目 孙 下 。 使 用 web 浏览 需 否 看 这 个 表单 ， 然 后 
笑 试 上 传 一 个 文件 。 如 果 成 功 了 ， 你 会 在 浏览 右 中 看 到 如 图 11-10 所 示 
的 结 末 。 


| a] localhost/11/do_upload.ph; x b= o o 


= C | © http://localhost/11/do_upload.php A, 


path: C:\Users\JM\AppData\Local\Temp\phpC6C3 tmp 
name: IMG 20101107 131718.jpg 

type: imagejpeg 

size: 960642 

File was moved! 





图 11-10 ”程序 清单 11.14 的 示例 结 
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事情 变 得 令 人 激动 起 来 。 当 然 ， 还 是 漏 反 了 少数 儿 项 ， 但 是 本 书 番 
下 的 部 分 将 继续 元 实 。 既 然 已 经 从 用 户 那 里 获取 了 信息 ， 如 末 能 够 使 用 
它们 做 一 些 事情 ， 例 如 ， 将 其 写 入 到 文件 ， 那 将 会 更 有 和 意思， 也 许 是 这 
样 吧 ? 这 融和 是 后 续 各 章 的 主题 。 


在 本 章 中 ， 我 们 学 习 了 如 何 使 用 各 种 超 全 局 变量 和 表单 输入 。 我 们 

还 学 习 了 如 何 回 客 户 姗 发 送 原始 的 标 头 以 重 定 同 用 户 。 我 们 学 习 了 如 何 
从 表单 提交 获取 列表 信息 以 及 如 何 使 用 隐 蔚 字段 在 脚本 调用 之 间 传 递 信 
息 。 最 后 ， 我 们 学 习 了 如 何在 Email 中 发 送 表单 结果 ， 以 及 如 何 使 用 一 
个 PHP 脚 本 通过 web 浏览 器 上 传 文件 。 


11.9 Q&A 


Q: 当 我 在 线 提 交 其 他 表单 的 时 候 ， 有 时 候 ， 我 会 看 所 输入 
的 值 位 于 导 同 下 一 个 页 面 的 URL 中 。 为 什么 会 这 样 ? 


A: 如 果 你 提交 一 个 表单 ， 例 如 一 次 Google 搜 索 ， 并 且 你 看 到 的 下 
一 个 URL 包 含 了 所 输入 的 值 ， 例 如 ， 搜 索 “cheese” 可 能 会 产生 如 下 所 示 
的 一 个 URL。 


https: //www, google.com/#hl=en&output=search&q=cheese 


然后 ， 你 将 会 看 到 使 用 GET 操 作 而 不 是 POST 操 作 的 一 个 表 早 的 输 
出 。 在 这 个 例子 中 ， 全 少 有 两 个 字段 ， 一 个 隐藏 子 段 ， 叫 做 “output”; 
为 一 个 是 可 以 看 到 的 字段 ， 叫 做 “gq”( 可 能 是 表示 query)。 


“cheese” 值 就 是 你 在 INPUT 框 中 输入 的 值 。 
Q: 为 什么 需要 在 一 个 表单 上 限制 上 传 大 小 。 


A: 如 果 在 设计 用 来 上 传 文件 的 表单 上 没有 限制 大 小 的 话 ， 你 将 会 
过 到 这 样 的 情况 ， 束 是 将 用 户 引 叶 到 一 个 无 法 完成 的 动作 ， 这 会 导致 他 
们 的 系统 和 你 的 系统 饭 冻 结 。 考 虑 下 这 样 的 情况 ， 当 你 想 要 接受 文件 或 
数字 疼 像 上 传 的 时 候 ， 用 户 创建 了 一 个 很 大 的 图 像 ， 假 设 是 1OMB。 如 
末 你 本 来 只 是 想 要 接受 岁 像 的 缩 略 网 ， 第 规 情 况 下 只 有 350kB 那 么 大 ， 
只 要 告诉 用 户 将 六 小 限制 在 这 个 范围 区 好 了 。 只 要 在 表单 中 组 合 
MAX_FILE_SIZE 和 upload_max_filesize 的 php.ini 设 置 ， 你 可 以 确保 单个 


H P EN BR YE AS 2 IENA 


11.10 KRAJ 


这 个 实践 练习 设计 用 来 帮助 你 预测 可 能 的 问题 、 复 习 已 经 学 过 的 知 
识 ， 并 且 开 始 把 知识 用 于 实践 。 


问答 题 


1. 用 来 找到 脚本 的 名 字 的 预定 义 变量 是 什么 ? 


2. 哪个 内 建 的 天 联 数 组 包含 了 作为 POST 请 求 的 一 部 分 提交 的 所 有 
HHE? 


3. 哪个 内 建 的 关联 数组 包含 了 作为 文件 上 传 的 一 部 分 提交 的 所 有 
的 值 ? 


A. 什么 阔 数 用 来 把 浏览 占 重 定 癌 到 一 个 新 的 页 面 ? 
5. mailO 函 数 所 使 用 的 4 个 参数 是 什么 ? 


6. 在 客户 凯 ， 如 何 限 制 一 个 用 户 通 过 一 个 特定 的 上 传 表 单 所 上 传 
的 文件 的 大 小 ? 


oy 


变量 $ SERVER[PHP_SELF] 保 存 了 脚本 的 名 字 。 
. $_POST 超 全 局 变量 。 

. $_FILES 超 全 局 变量 。 

. 带 有 一 个 Location 的 header(0) 函 数 。 


. 收 件 人 、 主 题 、 消 四 字 人 符 果 和 附加 的 标 头 。 


在 表单 中 使 用 一 个 名 为 MAX_FILE _ SIZE 的 隐藏 字段 。 


EA wel 


1 创建 一 个 计算 器 脚本 ， 它 允许 用 户 提交 两 个 数字 并 且 选 择 一 种 
对 这 两 个 数 执行 的 运算 (加 减 乘除 )， 

2， 在 思考 题 1 创建 的 脚本 中 使 用 隐藏 字段 ， 来 存储 和 显示 用 户 提交 
的 请 求 的 数字 。 


第 12 瘟 ”使 用 Cookie 和 用 户 会 话 


ERER, MÄ FI: 


。 如 何 存储 和 获取 cookie 信 息 。 
。 什么 是 会 话 变 量 以 及 它们 如 何 工 作 。 
。 如 何 开 始 或 继续 一 个 会 话 。 
。 如 何在 一 个 会 话 中 存储 变量 。 
。 如 何 销毁 一 个 会 话 。 
。 如 何 重 新 设置 会 话 变 量 。 
PHP 包含 了 很 多 的 函数 ， 可 以 用 来 管理 和 记录 用 刀 信 息 ， 包 括 简 单 


的 cookie 和 全 方位 的 用 户 会 话 。 会 话 使 用 PHP 语言 内 建 的 技术 ， 使 得 
保存 状态 束 像 是 引用 超 全 局 变量 那样 简 蛙 。 


12.1 Cookie 人 简介 


我 们 可 以 和 PHP 脚 本 一 起 使 用 cookie 来 存储 一 些 关 于 用 户 的 较 小 的 
信息 。cookie 是 由 用 户 浏 贤 右 存储 的 少量 数据 ， 它 和 一 个 来 和 目 服务 器 或 
AAMAS I ta RAS. WHI SA a, PS ESE EDLY DA ok 
你 和 存 20 个 cookie。 每 个 cookie 包 含 一 个 名 字 、 值 和 过 期 日 期 ， 以 及 主机 
和 路 径 信息 。 一 个 单个 的 cookie 的 大 小 限制 是 4kB。 


在 设置 了 cookie 之 后 ， 只 有 发 出 请 求 的 主机 能 够 读 取 数据 ， 这 束 保 
证 了 用 户 隐 私 得 到 卫 重 。 男 外 ， 用 户 可 以 配置 目 己 的 浏 史 絮 通 知 他 接受 
或 是 拒绝 所 有 cookie 的 请 求 。 因 此 ，cookie 应 该 适度 地 使 用 ， 并 用 在 没 
有 设计 事先 警告 用 户 的 一 个 环境 中 ， 不 应 该 作为 一 个 基本 元 系 而 依赖 。 


12.1.1 深入 了 解 一 个 cookie 


设置 一 个 cookie 的 PHP 脚 本 可 能 发 送 如 下 所 示 的 标 头 。 


HTTP/1.1 200 OK 

Date: Wed, 18 Jan 2012 10:50:58 GMT 

Server: Apache/2.2.21 (Unix) PHP/5.4.6 

X-Powered-By: PHP/5.4.@ 

Set-Cookie: vegetable=artichoke; path=/; domaLtn=yourdomain.com 
connection: close 

Content-Type: text/html 


正如 你 所 看 到 的 ，Set-Cookie 标 头 包含 了 一 个 名 / 值 对 、 一 个 路 径 和 
一 个 域 。 如 果 设 置 了 expiration 衬 段 ， 它 会 提供 浏览 奉 在 哪个 日 期 “二 
记 ”cookie 的 值 。 如 条 没有 设置 过 期 日 期 ， 当 用 户 会 话 过 期 的 时 候 ， 也 束 
是 当 用 户 关 挥 浏览 器 的 时 候 ，cookie 束 过 期 了 。 


path 字 段 和 domain 字 段 协 同 工 作 ， 因 为 path 十 找到 domain 的 一 个 月 
孙 ，cookie 应 该 送 回 给 服务 右 的 这 个 目录 下 面 。 如 末路 径 是 “”， 这 是 很 
种 见 的 什 ， 症 味 看 cookie 可 以 由 文档 根 目录 下 的 任何 文件 读 取 。 如 果 路 
径 是 “%products/”， 这 个 cookie 只 能 够 被 Web 站 点 的 /products 目 录 下 的 文件 
读 取 。 


domain 学 段 表 示人 允许 基于 cookie 的 通信 有 所 来 自 的 Internet 域 。 例 
如 ， 如 果 我 们 的 域 是 www.yourdomain.com， 并 且 使 用 
www.yourdomain.com 作 为 cookie 的 domain 值 ， 只 有 在 浏览 
www.domain.com 的 时 候 ，cookie 才 是 有 效 的 。 如 条 在 用 户 浏 览 的 过 程 
中 ， 我 们 发 送 给 用 户 诸 如 www2.domain.com 或 billing.domain.com 的 某 个 
域 ， 这 可 能 会 引 友 一 个 问题 ， 因 为 最 和 初 的 cookie 不 再 有 效 。 因 此 ， 在 
cookie 定义 中 的 domain 项 中 只 是 以 点 开头 ， 而 省 略 挥 主机 ， 例 
如 .domain.com， 这 种 方法 是 很 第 见 的 。 按 照 这 种 方法 ，cookie 将 会 对 充 
域 上 的 所 有 主机 都 有 效 。 域 不 能 和 cookie 所 有 发送 目的 域 不 同 ， 个 则 ， 
cookie 将 不 能 正确 工作 ， 束 算是 能 工作 ，Web 浏 贤 右 也 会 完全 拒绝 


cookie. 


12.1.2 ”访问 cookies 


如 果 Web 浏 览 器 配置 为 存储 cookie， 它 将 保持 基于 cookie 的 信息 直 
AIT HAA BH. WRH P EHN AA LAF cookie Hy AEA ERNE R 
面 ， 它 将 会 把 cookie 重 新 发 送 给 服务 器 。 浏 览 问 的 标 头 可 能 会 如 下 所 
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ZJN o 


GET / HTTP/1.@ 

Connection: Keep-Alive 

User-Agent: Mozilla/5.@ (Windows NT 6.1; WOW64} AppleWebKit/535.7 (KHTML, like 
Gecko) Chrome/16.0.912.75 Safar1/535.7 


Host: www. yourdomain.com 

Accept: image/gif, image/x-xbitmap, image/jpeg, image/pjoeg, image/png, */* 
Accept-Encoding: gzip 

Accept-Language: en,pdf 

Accept-Charset: 180-8859-1,*,utf-8 

Cookie: vegetable=artichoke 


随后 ， 一 个 PHP 脚 本 将 能 够 访问 cookie，cookie 在 环境 变量 
HTTP_COOKIE 中 或 者 作为 $_COOKIE 超 全 局 变量 的 一 部 分 ， 我 们 可 以 
用 如 下 3 种 方式 来 访问 它们 。 
echo $ SERVER['HTTP COOKIE']; // will print "vegetable=artichoke" 


echo getenv('HTTP_COOKIE'}; jf will print “vegetable=artichoke" 
echo $ COOKIE[ vegetable']; // will print "artichoke" 


12.2 ”使 用 PHP 议 四 一 个 cookie 


我 们 可 以 用 两 种 方法 在 一 个 PHP 脚 本 中 设置 一 个 cookie。 首 先 ， 用 
header() 函 数 来 设置 Set-Cookie 标 涉 。header(0) 函 数 需 要 一 个 字 从 串 ， 该 字 
符 串 随后 将 包含 到 服务 器 啊 应 的 标 头 部 分 。 由 于 标 头 会 为 你 自动 及 送 ， 
header0 必 须 在 友 送 给 浏览 右 的 任何 和 输出 之 前 调用 。 


header("Set-Cookie: vegetable=artichoke; expires=Thu, 19-Jan-12 14:39:58 GMT; 
path=/; domain=yourdomain.com') ; 


尽管 没什么 困难 ， 这 种 设置 cookie 的 方法 还 是 需要 我 们 编写 一 个 函 
数 来 构建 标 头 字符 串 。 像 这 个 例子 那样 格式 化 日 期 并 对 名 / 值 对 进行 
URL 编 色 并 不 是 特别 艰难 的 任务 ， 但 它 古 一 项 重复 性 的 工作 ， 因 而 PHP 
提供 了 一 个 函数 ， 这 了 吏 是 setcookie()。 


setcookie() 函 数 所 做 的 事情 就 像 它 的 名 字 所 显示 的 那样 ， 它 输出 一 
个 Set-Cookie 标 头 。 因 此 ， 写 应 该 在 任何 其 他 内 容 肥 送 给 浏 虎 项 之 前 调 
用 。 这 个 函数 接受 cookie 名 字 、cookie 值 、UNIX 时 间 惟 格式 的 过 期 日 
期 、 路 径 、 域 ， 以 及 一 个 整数 ， 如 果 cookie 仅 通过 一 个 安全 的 连接 及 送 
的 话 ， 这 个 整数 的 值 设置 为 1。 除了 第 一 个 参数 〈cookie 名 字 ) 以外， 
这 个 函数 的 所 有 参数 都 是 可 选 的 。 


程序 清单 12.1 使 用 setcookie0) 函 数 设 置 一 个 cookie。 


程序 清单 12.1 设置 并 显示 一 个 cookie 值 


1: <?php 

2: setcookie("vegetable", "artichoke", time(})+3600, "/", ".yourdomain.com", 6); 
GH 

4: if (isset($ COOKIE['vegetable'])) 1 

D echo "<p>Hello again! You have chosen: ".$ COOKIE['vegetable'].".</p>"; 
6: } else { 

T: echo "<p>Hello, you. This may be your first visit.</p>"; 

8: } 

as i= 


即便 我 们 在 脚本 第 一 次 运行 的 时 候 设 置 cookie〈 在 第 2 行 ) ， 
$ COOKIE [ 'vegetable' ] 变 量 也 不 会 在 这 时 候 创 建 。 由 于 只 有 当 浏 览 右 
将 一 个 cookie 发 送 到 服务 器 的 时 候 ， 才 会 读 取 它 ， 因 此 ， 直 到 用 户 重新 
访问 这 个 域内 的 一 个 页 面 的 时 候 ， 我 们 才能 够 读 取 它 。 


我 们 在 第 2 行 把 cookie 名 字 设 置 为 “vegetable”， 把 cookie 值 设置 
为 “artichoke”。 使 用 time() 函 数 来 获取 当前 的 时 间 惟 ， 并 且 在 其 上 添加 
3600 (一 小 时 有 3600 秒 ) ， 这 个 总 和 表示 我 们 的 过 期 日 期 。 我 们 定义 
了 一 个 路 径 “/”*， 表 示 一 个 cookie 应 该 为 服务 占 坏 境 下 的 所 有 页 面 发送 。 
我 们 把 domain 参 数 设 置 为 “.yourdomain.com”( 你 应 该 针对 自己 的 域 做 相 
应 的 修改 ， 或 者 使 用 localhost，， 这 意味 看 cookie 可 以 发 送 给 该 域 中 的 
任何 服务 器 。 最 后 ， 把 0 传递 给 setcookie()， 表 示 cookies 可 以 在 一 个 非 安 
全 的 环境 中 发 这 。 对 setcookie() 的 字符 串 参数 友 运 一 个 空 字 从 串 或 者 对 
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给 setcookie() 传 递 一 个 空 字符 串 作为 字符 串 参 数 或 0 作为 其 整数 字 
段 参数 ， 将 会 导致 忽略 这 些 参数 。 





通过 在 cookie 中 使 用 一 个 动态 创建 的 过 期 时 间 ， 如 程序 清单 12.1 所 示 ， 注 意 ， 过 期 时 间 是 
通过 在 运行 Apache 和 PHP 的 机 器 的 当前 系统 时 间 加 上 一 定 的 秒 数 而 创建 的 。 如 果 这 个 系统 时 


闻 不 是 准确 的 ， 有 可 能 它 发 送 一 个 已 经 过 去 的 过 期 时 间 到 cookie 中 。 





我 们 可 以 在 最 新 的 Web 浏 顺 郑 中 否 看 目 己 的 cookies 812-127; y 
程序 清单 12.1 中 存储 的 cookie 信 息 ，cookie 的 名 字 、 内 容 和 过 期 日 期 都 如 
期 出 现 。 当 我 们 在 目 己 的 域 上 运行 这 个 脚本 的 时 候 ， 域 名 可 能 不 同 。 





Sa 
[=] localhost/12/setcookie. php = ' 4, Options - Cookies and Oth: 
€ C |O chrome://settings/cookies wri’ 
Options Cookies and Other Data 
Site Locally Stored Data pers ea) | Ar localhost " 
Pr ca 
Basics localhost 1 cookie x 
Personal Stuff sheet 
: EGS Hood ay Name: vegetable 
ee Content: artichoke 
Pi Domain: localhost 
Path: / 7 
Im: send For: Any kind of connection 
Accessible to Script: Yes 
Created: Wednesday, January 16, 2012 10:57:11 AM 
Wi Expires: Wednesday, January 16, 2072 11:57:11 AM 
on Remove 
Jai 


要 了 解 使 用 cookie 的 更 多 信息 ， 尤 其 是 setcookie0O 函 数 的 信息 ， 


图 12-1 在 一 个 Web iil bias H AATF fia cookie 











考 位 于 http://www.php. net/setcookie 的 PHP 手 册 。 


删除 一 个 cookie 


正式 地 讲 ， 要 删除 一 个 cookie， 只 需 
setcookieO0， 示 例如 下 。 


要 调用 市 有 cookie 名 字 参 数 的 


setcookie(' vegetable"); 


然而 ， 这 种 方法 并 不 总 是 委 效 ， 并 且 不 能 依 顿 这 种 方法 。 相 反 ， 要 
删除 一 个 cookie， 使 用 一 个 确定 已 经 过 期 的 时 间 来 设置 cookie， 这 种 方 
法 是 最 安全 的 。 
setcookle{"vegetable’, "", time(}-60, "/", "yourdomain.com", O}; 


1s BE A PR TR A setcookie() -5 Hx 4) VW cookiet PTE H HY zeh E YY 
路 径 、 域 和 安全 参数 。 


12.3 A TE PA BUM w, 


ee 
取 连 接 到 该 标识 符 的 信息 。 当 一 个 访客 访问 一 个 支持 会 话 的 页 面 ， 
分 配 一 个 新 的 标识 符 ， 要 么 : in 
侍 重 新 关联 。 任 何 已 经 和 会 话 相 关联 的 变量 ， 孝 通过 $_SESSION EE 
局 变量 变 得 可 供 你 的 代码 使 用 。 


会 话 状态 通 各 存储 在 一 个 临时 文件 中 ， 尽 千 你 可 以 使 用 一 个 名 为 
Session_Sset_save_handler0 的 函数 实现 数据 库存 储 。 
session_set_save_handler0O 函 数 的 使 用 以 及 有 关 其 他 高 级 会 话 功能 的 讨论 
己 经 超出 了 本 书 的 范围 ， 但 是 ， 你 可 以 在 PHP 手 册 关 于 会 话 的 部 分 找到 

这 里 所 没有 讨论 的 项 目的 所 有 信息 。 


PAIT 2 


要 使 用 一 个 会 话 ， 我 们 需要 显 式 地 开始 或 继续 会 话 ， 除 非 我 们 已 经 
改变 了 php.ini 配 置 文件 。 默 认 情 况 下 ， 会 话 不 会 目 动 局 动 。 如 果 想 要 按 
这 种 方法 局 动 一 个 会 话 ， 我 们 必须 在 php.ini 文 件 中 找到 如 下 的 一 行 ， 将 
其 值 从 0 改 为 1， 并 且 重 新 启动 Web 服 务 器 


session.auto start = @ 


通过 把 session.auto_start 的 值 改 为 1， 我 们 可 以 确保 一 个 会 话 针 对 每 
个 PHP 文 档 而 局 动 。 如 条 你 不 想 改变 这 一 设置 ， 需 要 在 每 个 脚本 中 调用 


session_start() PAI ŽI- 


会 话 局 动 之 后 ， 我 们 立即 可 以 通过 session_id0) 函 数 访 问 用 户 的 会 话 
ID, session_id 0 函数 允许 你 设置 或 访问 一 个 会 话 ID。 程 序 清 单 12.2 FF 
始 一 个 会 话 并 且 将 会 话 ID 输出 到 浏览 器 。 


程序 清单 12.2 ”开始 或 继续 一 个 会 话 


: <?php 

: session Start( ) ; 

: echo "<p>Your session ID is ".session id().".</p>"; 
9% 


Wh — 


IK SASS UR A IS TT IN, ERTH 
Session_startO) 国 数 调 用 产生 一 个 会 话 ID。 如 果 这 个 脚本 稍 后 重新 载 入 或 
者 重新 访问 ， 同 一 个 会 话 ID 也 分 配给 该 用 户 。 这 个 操作 假设 该 用 户 已 经 
可 以 使 用 cookie 了 。 例 如 ， 当 我 们 第 一 次 运行 这 个 脚本 的 时 候 ， 输 出 如 
Pe 


Your session ID is 8joul7in51d@8e5onsjkblesié. 


当 我 们 重新 载 入 这 个 页 面 的 时 候 ， 输 出 仍然 如 下 。 


Your session ID is 8joul7ind51d@8eS5onsjkblesié6é. 
因为 我 们 已 经 可 以 使 用 cookie 了 并 且 会 话 ID 仍 然 存在 。 


当 第 一 次 人 切 始 化 一 个 会 话 的 时 候 ， 由 于 start_session0 笠 试 设置 一 个 
cookie， 因 此 必须 在 同 浏 览 器 输出 任何 内 容 之 前 调用 这 个 函数 。 如 果 没 
有 遵守 这 一 规则 ， 会 话 束 不 会 被 设置 ， 并 且 你 将 可 能 在 页 面 上 看 到 和 警 
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只 要 Web 浏览 右 是 激活 的 ， 驶 将 一 直 傈 持 当 前 会 话 。 如 采用 户 重 
新 打开 浏览 右 ，cookie 束 不 再 和 存储。 我 们 可 以 在 php.ini 文 件 中 修改 
Session.cOokie_jlifetime 设 置 ， 从 而 改变 这 一 行为 。 黑 认 值 是 0， 但 是 我 们 
可 以 设置 一 个 以 秒 为 单位 的 过 期 周期 。 


12.5 ”使 用 会 话 和 变量 


在 每 一 个 PHP 文 档 中 访问 一 个 唯一 的 会 话 标识 符 只 是 会 话 功能 的 开 
始 。 当 一 个 会 话 局 动 后 ， 我 们 可 以 在 超 全 局 变量 $_ SESSION 中 存储 任意 
多 个 变量 ， 然 后 在 任何 文 持 会 话 的 页 面 上 访问 它们 。 


程序 清单 12.3 向 超 全 局 变量 $_ SESSION 添加 了 两 个 变量 : product1 
和 product2〈 第 3 行 和 第 4 行 )。 


程序 清单 12.3 ”在 一 个 会 话 中 存储 变量 


1: <?php 

2: session startt{); 

3: $ SESSION[ 'producti'] = "Sonic Screwdriver"; 
4: $ SESSION[ 'product2'] = "HAL 2000"; 

5: echo "The products have been registered."; 
5: «> 


在 用 户 移动 到 另 一 个 新 的 页 面 之 前 ， 程 序 清单 12.3 中 的 神奇 之 处 不 
会 体现 出 来 。 程 序 清 单 12.4 创 建 了 一 个 单独 的 PHP 脚 本 ， 这 个 脚本 访问 
存储 在 超 全 局 变量 $ SESSION 中 的 变量 。 


程序 清单 12.4 访问 存储 的 会 话 变量 


: <?php 

: session start(); 

?> 

<p>Your chosen products are:</p> 

<ul> 

: <li><?php echo $ SESSION[ product1 ] ; ?></1i> 
: <li><?php echo $ SESSION['product2']; ?></1i> 
: </ul> 


OnNOoah WDD — 


图 12-2 显示 了 来 目 程序 清单 12.4 的 输出 。 正 如 你 所 见 到 的 ， 我 们 


已 经 在 一 个 全 新 的 页 面 中 访问 了 $4_SESSION [ 'product1' ]4#/1$_ SESSION [ 
product2' ] 变 量 。 





[=] localhost/12/getfromsessic 
a C © http://localhost/12/getfromsession.php x, 
Your chosen products are: 


* Sonic Screwdriver 


+ HAL 2000 


图 12-2 访问 存储 的 会 话 变 量 


管 不 是 一 个 非 章 有趣 或 有 用 的 例 玫 ， 但 这 个 脚本 确实 展示 了 如 何 
pa 在 项 后 ，PHP 把 信息 号 入 到 一 个 临时 文件 ， 可 以 
wiji session_save_path() KAEA XA ES R T RRETAN 

图 数 可 选 地 接受 到 一 个 目录 的 路 径 ， 并 且 把 所 有 的 会 话 文件 都 写 入 
aie, 如 采 我 们 不 传递 给 它 参 数 ， 它 返回 一 个 字符 串 ， 表 示 会 话 文件 保 
存 的 当前 目录 。 在 我 的 系统 上 执行 如 下 语句 。 


echo session save path(}; 


会 显示 出 一 个 /tmp。 看 一 下 我 的 /tmp 目 录 会 显示 出 很 多 文件 ， 其 名 
字 如 下 所 示 。 


sess fa963e3e49186764b021 Be82d050de7b 
sess 76cae8aci23ibilatazc69935c11ddg95 
sess bb50771a769c605a077424d59c 784eae@ 


当 我 第 一 次 运行 程序 清单 12.2 的 时 候 ， 打 开 和 所 分 配 的 会 话 ID 相 
匹配 的 文件 ， 可 以 看 到 已 注册 的 变量 是 如 何 存储 的 。 


Product11sS:17:"Sonie Screwdriver";product2|s:8:"HAL 2006"; 


当 一 个 值 放置 在 $_ SESSION 超 全 局 变量 中 ，PHP 把 变量 名 和 值 与 入 
到 一 个 文件 中 。 正 如 我 们 所 看 到 的 ， 这 个 信息 可 以 读 取 并 且 变 量 可 以 稍 
后 恢复 。 当 我 们 把 这 个 变量 添加 到 超 全 局 变量 $_ SESSION 后 ， 你 仍然 可 
以 在 脚本 执行 过 程 中 的 任何 时 刻 修改 其 值 ， 但 是 ， 这 个 修改 后 的 值 不 会 
反映 到 全 局 设置 中 ， 直 到 把 这 个 变量 重新 赋值 给 超 全 局 变量 
$ SESSION. 


Re ris £212.37 RATER EREA BE AES SESSION 
的 过 程 。 然 而 ， 这 个 例子 并 不 是 非 间 灵活。 理想 情况 下 ， 你 所 能 注册 的 
值 的 数目 应 该 是 可 变 的 。 例 如 ， 可 能 硕 户 用户 从 一 个 列表 中 选择 产品 。 
在 这 种 情况 下 ， 可 以 使 用 serialize() 函 数 来 把 数组 存储 到 会 话 中 。 


程序 清单 12.5 创 建 了 一 个 表单 ， 它 允许 一 个 用 户 来 选择 多 个 产品 。 
我 们 可 以 使 用 会 话 变量 来 创建 一 个 基本 的 购物 车 。 


程序 清单 12.5 ”把 一 个 数组 变量 添加 到 一 个 会 话 变 量 中 


I <7 OND 

2: session start(); 

3: ie 

4: <!DOCTYPE html> 

5: <html> 

6: <head> 

7: <title>Storing an array with a session</title> 

8: </head> 

9: <body> 

10: <h1i>Product Choice Page</h1> 

11: <?php 

12: if (isset($ POST['form products']}) { 

is: if (!empty($ SESSION[ 'products'])}) { 

14: $products = array unique ( 

La array _merge(unserialize($ SESSION[ 'products']), 
16: $ POST['form products'])); 

1 $ SESSION[ 'products'] = serialize($products) ; 
183 } else { 

19: $ SESSION[ 'products'] = serialize($ POST['form_products']); 
20: } 
reo echo "<p>Your products have been registered!</p>"; 
220 g 
2a: T> 


24: <form method="post" action="<?php echo $ SERVER['PHP SELF']; ?>"> 

25: <p><label for="form_products">Select some products:</label><br /> 

26: <select id="form products" name="form products[]" multiple="multiple" 
$ize="3"> 

27: <option value="Sonic Screwdriver">Sonic Screwdriver</option> 

28: <option value="Hal 2000">Hal 2000</option> 

29: <option value="Tardis">Tardis</option> 

30: <option value="ORAC">ORAC</option> 

31: <option value="Transporter bracelet">Transporter bracelet</option> 

32: </select></p> 

33: <button type="submit" name="Submit" value="choose">Submit Form</button> 

34: </form> 

35: <p><a href="sessionl.php'">go to content page</a></p> 

36: </body> 

37: </html> 


我 们 通过 在 第 2 行 调 用 session_start0) 来 司 动 或 继续 一 个 会 话 。 这 个 
调用 使 我 们 能 够 访问 之 前 设置 的 任何 会 话 变 量 ， 第 24 行 开始 一 个 HTML 
表单 ， 在 第 26 行 创建 了 一 个 名 为 form_products [] 的 SELECT 元 素 ， 其 中 
包含 了 多 个 产品 的 OPTION 元 系 。 


加 


别 生 了 ，HTML 的 表单 元 素 允 许 拥 有 方 括号 的 多 个 选项 附加 到 它们 的 NAME 参 数 的 值 之 





后 。 这 束 使 得 用 尸 的 选择 可 以 在 一 个 数组 中 使 用 。 





在 第 11 行 开始 的 PHP 代 码 块 中 ， 我 们 测试 了 $_POST [ 
'form_products' ] 数 组 的 存在 《第 12 行 ) 。 如 果 这 个 变量 存在 ， 我 们 可 
以 假设 表单 已 经 提交 并 且 信 息 己 经 存储 到 超 全 局 变量 $ SESSION 中 。 


随后 ， 在 第 12 行 测试 一 个 叫做 $_ SESSION [ 'products' ] 的 数组 。 如 
果 这 个 数组 存在 ， 之 前 访问 这 个 脚本 的 时 候 填 元 过 该 数组 ， 这 样 我 们 刺 
可 以 将 其 和 $_POST['form_products' ] 数 组 合并 ， 提 取出 唯一 的 元 系 ， 并 
且 把 结果 赋值 回 $products 数 组 〈 第 14 行 到 第 16 行 ) 。 随 后 ， 在 第 17 行 把 
$products 数 组 添加 到 $_SESSION 超 全 局 变量 。 


第 35 行 包含 了 到 其 他 脚本 的 一 个 链接 ， 我 们 将 用 这 个 脚本 展示 对 
用 户 选 取 的 产品 的 访问 。 我 们 在 程序 清单 12.6 中 创建 了 这 个 脚本 ， 同 
时 把 程序 清单 12.5 中 的 代码 保存 为 arraysession.php。 

来 看 看 程序 清单 12.6 是 如 何 访 问 存 储 到 会 话 中 的 那些 项 的 ， 注 意 会 
话 是 在 arraysession.php 中 创建 的 。 


程序 清单 12.6 ”访问 会 话 变量 


<?php 

session start({); 

T> 

<!DOCTYPE html> 

<html> 

<head> 

<title>Accessing session variables</title> 
</head> 

9: <body> 

1@: <hi>Content Page</h1> 

11: <?php 

12: if (isset($ SESSION[ 'products'])) { 

13: echo "<strong>Your cart:</strong><ol>"; 
14: foreach (unserialize($ SESSION['products]) as $p) { 
153 echo "<li>" Sp "<li>: 


O N Oo oh 一 


ie echo "</ol>"; 
18: } 
19: 7> 


20: <p><a href="arraysession.php">return to product choice page</a></p> 
21: </body> 


22: </html> 


再 一 次 ， 我 们 在 第 2 行使 用 session_start0) 来 继续 会 话 ， 在 第 12 行 
测试 了 $_SESSION [ products' ] 变 量 的 存在 性 。 如 果 和 存在， 就 会 将 它 肥 
序列 化 ， 并 且 在 第 14 行 到 第 16 行 过 历 它 ， 在 浏览 右 显 示 出 每 个 用 户 的 
选择 项 。 图 12-3 给 出 了 这 个 例子 的 运行 效果 。 





Accessing session variables 


个 C © hitp://localhost/12/session1.php a, 
Content Page 
Your cart: 

1. Tardis 

2. ORAC 


return to product choice page 


图 12-3 访问 会 话 变 量 的 一 个 数组 


当然 ， 对 于 一 个 其 正 的 购物 车 程序 ， 我 们 应 该 把 产品 细 市 你 存在 一 
个 数据 库 中 并 且 测 话 用 户 输 入， 而 不 是 让 目地 存储 和 旺 示 它 。 但 是 ， 程 
序 清单 12.5 和 程序 清单 12.6 展 示 了 我 们 可 以 很 容易 地 使 用 会 话 函 效 来 访 
问 其 他 页 面 中 设置 的 数组 变量 。 


12.6 HRZ 8 A Be ee 


FATT AY LME Ad session_destroy(SK 24 R— The ik, YARRA Emn 
量 。session_destroyO 图 数 不 需 要 参数 ， 应 该 有 一 个 已 经 建立 的 会 话 供 这 
个 函数 操作 。 如 下 的 代码 段 首 先 继续 一 个 会 话 然后 销毁 它 。 


session start(); 
session destroy(}; 


当 我 们 移动 到 使 用 一 个 会 话 的 其 他 页 面 的 时 候 ， 已 经销 毁 的 会 话 将 
不 能 再 使 用 ， 这 迫使 它们 局 动 目 己 的 新 的 会 话 。 任 何 已 注册 的 变量 将 会 
> 


然而 ，session_destroyO 函 数 不 会 立刻 销毁 已 注册 的 变星。 对 于 那些 
在 其 中 调用 了 session_destroy() 的 脚本 ， 它 们 仍然 可 以 访问 这 些 变 量 和 直 
到 变量 被 重新 载 入 。 下 面 的 代码 段 首 先 继 续 或 者 局 动 一 个 会 话 ， 并 且 注 
有 册 一 个 名 为 test 的 变量 ， 这 个 变量 的 值 我 们 设置 为 5。 销 开 这 个 会 话 并 不 
会 销毁 这 个 已 注册 的 变量 。 
session_start(); 
$ SESSION['test'] = 5; 


session destroy(); 
echo $ SESSION[ test ]; // prints 5 


BN — Serta PIM BR ATA Eee, JA rs Ee Te ee, W 
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session start(}; 

$ SESSION['test'] = 5; 

session destroy({); 

unset($ SESSION[ 'test']); 

echo $ SESSION['test']; // prints nothing (or a notice about an undefined index) 


12.7 在 一 个 珊 有 注册 用 户 的 环境 中 使 用 会 话 


到 现在 所 见 到 的 例子 对 于 会 话 还 都 只 是 浅 答 辑 止 ， 但 是 ， 可 以 这 么 
说 ， 可 能 需要 一 些 附加 说 明 来 防止 会 话 的 滥用?"。 下 面 的 两 个 小 节 给 出 
了 第 见 会 话 用 法 的 一 些 例子 。 在 本 书 和 后 的 各 章 中 ， 会 证 将 用 于 我 们 将 
要 构建 的 一 个 示例 应 用 程序 。 


12.7.1 使 用 注册 的 用 户 


假设 你 已 经 创建 了 一 个 在线 社区 或 门户 网 站 ， 或 者 是 用 户 可 以 加 入 
的 茶 种 其 他 类 型 的 应 用 ， 这 个 过 程 通 币 包含 一 个 注册 表单 ， 用 户 在 其 中 
创建 一 个 用 户 名 和 密码 并 且 质 写 一 个 个 人 身份 表格 。 从 这 个 页 面 肥 大 的 
那 一 刻 开始 ， 每 次 一 个 注册 的 用 户 登 录 到 系统 的 时 候 ， 我 们 都 可 以 抓 取 
用 户 的 号 份 信息 并 且 将 其 存储 到 一 个 用 户 会 话 中 。 


> 小 


你 决定 要 存储 到 一 个 用 户 会 话 中 的 项 目 应 该 十 这 样 的 项 ， 我 们 可 以 
想象 得 到 ， 这 些 项 会 用 得 很 少 ， 并 且 不 断 地 从 数据 库 中 提取 它们 的 效率 
很 牵 。 例 如 ， 假 设 我 们 创建 了 一 个 门 尸 网站， 其 中 的 用 户 会 分 配 一 个 级 
别 ， 例 如 管理 员 、 注 册 用 户 、 匿 名 访客 等 等 。 在 我 们 的 显示 模式 中 ， 应 
该 忌 是 要 检 权 验证 用 尸 所 访问 的 模块 是 侍 对 他 有 相应 的 许可 。 因 
此 ,“ 用 户 级 别 ? 应 该 是 存储 在 用 户 会 话 中 的 一 个 值 ， 所 以 ， 在 显示 请 求 
的 模块 中 所 用 到 的 身份 验证 脚本 只 需要 去 检查 一 个 会 话 变 量 ， 驳 设 必 要 
连接 到 数据 库 并 租 询 数据 了 。 


12.7.2 HHF tim 


如 朱 你 想 在 一 个 基于 用 户 的 应 用 程序 的 设计 阶段 感受 新 湖 ， 可 以 构 
建 这 样 一 个 系统 ， 让 注册 用 户 可 以 设置 具体 的 俩 好 来 影响 他 浏览 你 的 站 
氮 的 方式 。 例 如 ， 可 以 允许 用 户 从 一 个 预先 确定 的 其 色 方 案 、 字 体 字 所 
等 等 中 做 出 选择 。 或 者 ， 我 们 可 能 会 允许 用 户 打 开 或 关闭 对 于 东 一 组 内 
容 的 浏览 。 


这 些 功能 元 系 中 的 每 一 个 都 应 该 存储 到 一 个 会 话 中 。 当 用 户 登 录 ， 
这 个 应 用 程序 应 该 把 所 有 相关 的 值 都 载 入 到 该 用 户 的 会 话 中 ， 并 且 据 此 
来 对 后 续 请 求 的 每 一 个 页 面 做 出 反应 。 这 个 用 户 可 以 修改 其 俩 好 ， 他 可 
以 在 登录 的 时 候 做 到 这 一 点 ， 我 们 其 至 应 该 根据 存储 在 会 话 中 的 项 目 来 
预 乞 妆 载 一 个 “ 偶 好 ?表单 ， 而 不 是 回 到 数据 库 来 获取 这 些 项 目 。 如 采访 
用 户 在 登录 的 时 候 修改 了 任何 俩 好 ， 只 要 便 用 新 的 选项 来 秋 换 
$_SESSION 超 全 局 变量 中 的 值 束 可 以 了 ， 不 角 要 迫使 用 户 退 出 然后 册 次 
重新 登录 。 


P28 2H 


在 本 章 中 ， 我 们 学 习 了 在 一 个 无 状态 的 协议 中 保存 状态 的 不 同方 
法 ， 包 括 设置 一 个 cookie 和 启动 一 个 会 话 。 所 有 这 些 保存 状态 的 方法 都 
使 用 了 某 种 方式 的 cookie 或 者 查询 字符 串 ， 有 时 候 会 结合 使 用 到 文件 或 
数据 库 。 这 些 方法 都 各 有 优 缺 点 。 


我 们 了 解 了 一 个 cookie 本 质 上 不是 可 徘 的 ， 并 且 不 能 存储 太 多 的 信 
为 一 方面 ， 它 可 以 维持 一 个 较 长 的 时 间 。 把 信息 写 入 文件 或 数据 库 


a. 
AS PBR ERIK, HHE -TARA ET Rae Gell, BOER AR 
统 官 理 相 天 的 一 个 问题 。 


关于 会 话 目 对 ， 我 们 学 习 了 如 何 使 用 session_start() 局 动 或 继续 一 个 
会 话 。 当 处 于 一 个 会 话 中 时 ， 我 们 学 习 了 如 何 同 $_SESSION 超 全 局 变量 
洪 加 和 变量， 检查 其 存在 性 ， 如 有 果 和 需要 的 话 重 置 它们 ， 以 及 销 蝶 整个 会 


UE o 


12.9 Q&A 


4 


如 东 用 户 不 能 够 使 用 cookie 的 话 ， 应 用 程序 将 会 友 生 什么 
HAL? 


A: HWEL. BRD AREA Rki F cookiejt AA SS 
cookie， 应 用 程序 将 无 法 工作 。 然 而 ， 可 以 通过 声明 你 要 使 用 cookie 
告诉 用 户 开 司 cookie; 并 且 ， 在 对 应 用 程序 做 任何 “重要 的 ?事情 之 表 ， 
也 要 检查 cookie 十 个 可 以 使 用 。 当 然 ， 这 么 做 的 意 风 是， 即便 用 户 忽 略 
了 你 为 了 使 用 应 用 程序 必须 打开 cookie 的 提示 ， 如 果 cookie 测 斌 失效 了 
而 导致 不 允许 用 户 执 行 一 个 操作 ， 这 也 会 唤起 用 尸 的 注意 。 


Q: 我 应 诅 知 道 到 会 话 函 数 的 所 有 和 缺 点 吗 ? 


A: Sia SUH A Æ. ZAIN, Fits S, cookie 不 能 跨越 多 个 
SEAM, AUC, WRM fel] —TS IRA EHAA CH EN 
个 电子 商务 环境 的 一 部 分 ) ， 你 可 能 需要 考虑 通过 在 php.ini 文 件 中 把 
session.use_cookies 指 令 设置 为 0 来 天 闭会 话 的 cookie。 


12.10 ”实践 练习 


这 个 实践 练习 设计 用 来 帮助 你 预测 可 能 的 问题 、 复 习 已 经 学 过 的 知 
识 ， 并 且 开 始 把 知识 用 于 实践 。 


1， 哪 个 函数 用 来 以 一 个 PHP 脚 本 启动 或 继续 一 个 会 话 ? 
2， 哪 个 函数 可 以 返回 当前 会 话 的 ID? 


3. 如 何 终 结 一 个 会 话 并 且 删 除 对 其 访问 的 所 有 记录 ? 


oy 


fie A 
1. 我 们 可 以 在 脚本 中 使 用 session_startO 函 数 来 局 动 一 个 会 话 。 
2. 我 们 可 以 使 用 session_id0 函 数 来 访问 会 话 的 ID。 


3. session_destroy0) 函 数 删 除 一 个 会 话 及 请 求 的 所 有 记录 。 


AG el 


1. GSE“ AAAS, CEH A m PR BOR TRE PRIA R H A Wy 
了 哪些 页 面 。 


2. 创建 一 段 新 的 脚本 ， 它 整 页 整 页 地 列 出 在 你 的 环境 中 访问 的 用 
户 ， 以 及 他 们 访问 的 时 间 。 


功能 。 


单 。 
们 也 
示 如 


第 13 章 ”使 用 文件 和 目录 


ERER, MÄ FI: 


如 何 把 其 他 文件 包含 到 你 的 文档 中 。 
如 何 测试 文件 和 目录 的 存在 性 。 

如 何在 使 用 文件 前 打开 它 。 

如 何 从 文件 读 取 数据 。 

如 何 写 入 一 个 文件 或 回 其 添加 内 容 。 

如 何 锁定 一 个 文件 。 

如 何 使 用 目录 。 

如 何 把 数据 从 外 部 应 用 程序 导入 和 导出 。 
如 何 发 送 shell 命 令 并 且 在 浏览 问 显 示 结 


测试 文件 的 存在 性 以 及 读 写 文 件 ， 对 于 定编 程 语言 来 说 是 很 重要 的 
PHP 也 不 例外 ， 因 此 ， 它 提供 了 很 多 函数 使 得 这 一 过 程 相当 入 

另外 ， 由 于 PHP、Apache 和 MySQL 不 是 机 器 上 仅 有 的 应 用 程序 ， 我 
可 能 需要 在 PHP 代 但 中 访问 其 他 的 应 用 程序 。 本 草 中 的 元 系 回 你 展 
何 从 PHP 代 码 访问 其 他 的 应 用 程序 。 


13.1 使 用 include 语 名 包含 文件 


include 语 句 使 得 我 们 能 够 把 其 他 文件 ， 通 香 是 其 他 PHP 脚 本 包 侣 到 
PHP 文 档 中 。 这 些 被 包含 的 文件 中 的 PHP 代 但 惑 好 像 是 主 文 档 的 一 部 分 
一 样 来 执行 ， 这 对 于 在 多 个 页 面 中 包含 代码 库 是 很 有 用 的 。 


即便 我 们 创建 了 一 个 真正 有 用 的 函数 ， 如 果 没 有 include 语 句 ， 我 们 
到 现在 还 只 能 选择 将 函数 烙 贴 到 需要 使 用 它 的 每 个 文档 中 。 如 果 肥 现 函 
数 有 一 个 漏 调 bug) 或 者 想 要 添加 一 项 功能 ， 我 们 必须 找到 将 其 贴 入 
的 每 个 页 面 并 且 修 改 它 ， 一 次 叉 一 次 地 重复 。Imclude 语 句 使 我 们 不 必 这 
么 索 珊 。 我 们 可 以 把 新 创建 的 函数 放 加 到 一 个 单独 的 文档 中 ， 例 如 
myinclude.php， 然 后 ， 在 运行 的 时 候 ， 将 其 谈 入 到 需要 它 的 任何 页 面 。 


Include 语 句 需 要 一 个 参数 : 指 回 要 包含 的 文件 的 相对 路 径 。 程 序 清 

单 13.1 创建 了 一 个 简单 的 PHP 脚 本 ， 这 个 脚本 使 用 ipclude 语 句 来 包含 一 
个 文件 ， 并 且 输 出 该 文件 的 内 容 。 

程序 清单 13.1 使 用 include0) 


1: <?php 
2: include ‘myinclude.php' ; 
ay F> 


程序 清单 13.1 中 的 include 语 句 包 仿 了 文档 myinclude.php， 我 们 可 以 
在 程序 清单 13.2 中 看 到 被 包含 文档 的 内 容 。 
程序 清单 13.2 ”被 包含 到 程序 清单 13.1 中 的 文档 


1: I have been included!! 


把 程序 清单 13.1 的 内 容 放 入 到 一 个 名 为 test_include.php 的 文件 ， 并 
且 把 程序 清单 13.2 的 内 容 放 入 到 一 个 名 为 myinclude.php 的 文件 。 把 这 两 
个 文件 痢 放 到 Web 服 务 占 文档 根 目录 下 ， 当 我 们 使 用 Web 浏 唤 器 访问 
test_include.php 的 时 候 ， 将 会 输出 如 下 信息 。 


I have been included!! 


我 们 在 一 个 PHP 代 码 块 中 包含 了 纯 文本 ， 这 个 结果 对 你 来 说 看 上 去 
Aare. KME, DRES LFA BEREAN ACA, URE 
在 一 个 被 包含 文件 中 执行 PHP 代 码 ， 必 须 使 用 PHP 开 始 和 结束 标记 将 其 
包围 起 来 。 在 程序 清单 13.3 中 ， 我 们 修改 了 myinclude.php 的 内 容 ， 以 便 
代码 在 一 个 被 包含 文件 中 执行 。 


程序 清单 13.3 包含 PHP 代 码 的 一 个 包含 文件 


1: <?php 

2: echo "I have been included!!<br/>"; 

3: echo "But now I can add up... 4+ 4= ".(4 + 4); 
4: ?> 


把 程序 清单 13.3 的 内 容 放 入 到 一 个 名 为 myinclude2.php 的 文件 
中 ， 并 且 修 改 包 含 在 test_include.php 中 的 被 包含 文件 的 名 ， 让 它 指 同 这 
个 新 的 文件 。 把 这 些 文件 都 放 到 Web 服 务 咒 文档 根 目录 下 。 现 在 ， 当 我 
们 使 用 web 浏览 器 来 访问 test_include.php 的 时 候 ， 输 出 如 下 所 示 。 


I have been included! ! 
But now I can add up... 4+ 42: 8 


我 们 能 看 到 数字 8 的 唯一 途径 是 ， 把 4 和 4 相 加 的 代码 作为 PHP 执 
行 ， 并 且 事 实 正 是 如 此 。 


13.1.1 ”从 一 个 被 包含 文档 返回 一 个 值 


PHP PAL CTF AY DAR R aK I] — “Meo BIR FE PA — 
样 ， 在 被 包含 文件 的 末尾 使 用 return 语 句 来 结束 代码 的 执行 。 此 外 ， 不 
包含 额外 的 HTML。 在 程序 清单 13.4 和 程序 清单 13.5 中 ， 我 们 包含 了 
一 个 文件 并 且 将 其 返回 信 骨 给 一 个 变量 。 


程序 清单 13.4 ”使 用 include 来 执行 PHP 并 且 赋 返回 值 


: <?php 
: $addResult = include ‘returnvalue.php' ; 
: echo "The include file returned ".$addResult; 
?> 


Wh 


程序 清单 13.5 一 个 返回 一 个 值 的 包含 文件 


: <?php 
: $retval = (4 + 4 ); 
: return $retval; 
?> 
: This HTML will never be displayed because it comes after a return statement! 


Jo Ae ON =| 


把 程序 清单 13.4 的 内 容 放 入 到 一 个 名 为 test_returnvalue.php 的 文件 
中 ， 把 程序 清单 13.5 的 内 容 放 入 到 一 个 名 为 returnvalue.php 的 文件 中 ， 
把 这 些 文件 部 放 入 到 Web 服 务 右 文档 根 目录 中 。 当 我 们 通过 Web 浏 哆 可 
访问 test_returnvalue.php 的 时 候 ， 输 出 如 下 所 示 。 


The include file returned 8, 


正如 程序 清香 13.5 第 5 行 的 字符 串 所 示 ， 如 来 在 被 包含 文件 中 有 一 
个 PHP 语 句 块 的 证 ，PHP 语 句 块 以 外 的 任何 内 容 部 人 不 会 显示 。 


13.1.2 ”在 控制 结构 中 使 用 include 语 名 


如 末 我 们 想 要 在 一 个 条 件 语 句 中 使 用 include 语 可， 它 将 当 作 任何 其 
他 代码 一 样 处 理 ， 只 有 条 件 满足 的 时 候 ， 才 会 包含 文件 。 例 如 ， 如 下 代 


但 段 中 的 include 语 句 束 不 会 调用 。 


$test = "1"; 
1f ($test == "2") { 
include ‘file.txt ; // won't be included 


} 


如 果 我 们 在 一 个 循环 中 使 用 一 条 include 语 句 ， 那 么 每 次 循环 迭代 的 
时 候 都 将 包含 文件 一 次 。 程 序 清 单 13.6 通 过 在 一 个 for 循 环 中 使 用 一 条 
include 语 句 说 明了 这 一 概念 。 对 于 每 次 迭代 ，include 语 句 都 引用 一 个 不 
同 的 文件 。 


程序 清单 13.6 ”在 一 个 循环 中 使 用 include 


: <?php 

: for ($x = 13 $x<=3; Sxtt) { 

SLNCTLIS = inNGTLLe ebx txt 

echo "Attempting to include ".$incfile."<br/>'; 
include $incfile; 

echo "<hr/>"; 


co N 中 加 上 和 一 


把 程序 清单 13.6 的 内 容 保存 到 一 个 名 为 loopy_include.php 的 文件 
中 ， 并 且 将 其 放置 到 Web 服 务 右 的 文档 根 目录 下 ， 同 时 一 起 放置 的 还 有 
其 他 3 个 不 同 的 文件 incfilel.txt、incfile2.txt 和 incfile3.txt。 假 设 这 些 文件 
中 的 每 一 个 只 是 简单 地 包含 自己 的 文件 名 的 一 个 确认 ， 那 么 输出 将 会 如 
图 13-1 所 示 。 


= a a 


[=] localhost/13/loopy_include 
i C | © hitp://localhost/13/loopy_include.php T 


Attempting to include mchle l txt 
I am inchle 1 tat! 


2 ee ee 





Attempting to include mchile2 txt 
I am mehile2 txt! 


Attempting to inchide inchle3 txt 
I am mehile3 txt! 


oo EE E E A E E E E E E E E a E E E E E E E E E 





图 13-1 loopy_include.php Hy 447 14 


尽管 这 段 代 人 码 工 作 得 很 好 ， 但 一 些 情况 下 ， 按 照 这 种 方式 使 用 
include 语 句 并 不 是 一 个 好 的 想法 ， 我 们 将 在 下 一 节 中 看 到 。 


13.1.3 ”使 用 include once 语句 


在 代码 中 使 用 几 个 人 不同 的 代码 库 所 引起 的 问题 之 一 ， 束 是 存在 同一 
个 文件 个 两 次 调用 include 的 风险 。 这 有 时 候 会 及 生 在 较 大 的 项 目 上 ， 即 
当 不 同 的 库 文件 在 一 个 共同 的 文件 上 调用 include 的 时 候 。 两 次 包含 同一 
DL. ERTER Ps PASS, Ama AQPHP 9] SEW ot o 


RW (Wd include_oncei® ARAR B includetS J A) VAR ROK PETE Al 
include_once 语 句 需 要 一 个 包 人 文件 的 路 径 ， 此 外 ， 第 一 次 调用 它 的 情 


况 和 include 语 句 是 完全 相同 的 。 然 而 ， 如 果 在 脚本 执行 中 对 同一 个 文件 
册 次 调用 include_once 语 句 ， 这 个 文件 不 会 被 再 次 包 合 。 这 使 得 
include_once 语 句 成 为 创建 可 重用 的 代码 库 的 一 个 优秀 工具 。 


13.1.4 include_path 命 令 


使 用 ipclude0 和 include_once0 来 访问 代码 库 可 以 增加 项 目的 灵活 性 
和 可 重用 性 。 然 而 ， 还 是 有 一 些 头痛 的 问题 需要 兄 服 。 例 如 可 移 和 桓 性 ， 
如 来 你 下 接 编 码 包含 的 文件 的 路 径 ， 将 会 为 此 痛 吾 人 不堪。 假设 创建 了 一 
A lib 目录 并 且 通 过 目 己 的 项 目 以 如 下 方式 来 引用 它 。 


include_once '/home/user/bob/htdocs/project4/lib/mylib.inc.php'; 


当 把 目 己 的 项 目 移 动 到 一 个 新 的 服务 硕 时 ， 如 采 包 售 路 径 在 一 百 个 
REESE Pape BIST, PARES RMD MAEM AT 
甚至 更 多 的 包含 路 径 。 可 以 通过 在 php.ini 文 件 中 设置 mclude_path 命 令 来 
et FX — FUL o 


include path .:/fhome/user/bob/htdocs/project4/lib/ 


include_path (A JOUER eA, PHH Shae CE 
Windows 中 是 分 写 ) 。include_path 命 令 中 项 目的 顺序 决定 了 和 奏 找 指定 的 
文件 目录 的 顺序 。 第 一 个 冒号 前 的 第 一 个 点 (.) 表 示 “ 当 前 目录 ”， 默 认 应 
该 给 出 。 现 在 ， 在 include 或 include_once 中 使 用 的 任何 路 径 都 是 相对 于 
include_path 的 值 ， 示 例如 下 。 


include once ‘mylib.inc.php' ; 
当 转 到 目 己 的 项 目 时 ， 你 只 需要 修改 include_path 命令 。 


PHP 也 有 一 个 require 语 句 ， 它 执行 和 include 语 句 类 似 的 功能 ， 男 外 还 有 一 个 require_ once 
语句 。 不 溃 脚 本 的 流程 如 何 ，require 语 句 所 市 入 到 代码 中 的 一 切 内 容 都 会 执行 ， 因 此 ， 不 应 该 
将 其 用 作 条 件 或 循环 结构 的 一 部 分 。 另 外 要 注意 ， 作 为 一 条 require 语 句 的 结果 所 包含 的 文件 不 
RER |B] —ME 





13.2 ”验证 文件 


在 代码 中 使 用 一 个 文件 或 目录 之 前 ， 对 它 有 个 详细 了 解 是 一 个 不 错 
的 想法 ， 并 且 知 道 它 是 个 真 的 存在 是 一 个 不 错 的 开始 。PHP 提 供 了 很 多 
的 函数 来 帮助 发 现 关 于 你 的 系统 上 的 文件 的 信息 。 本 贡 人 简短 地 介绍 一 些 
最 有 用 的 函数 。 

13.2.1 使 用 file_exists0) 检 查 文 件 的 存在 性 

我 们 可 以 使 用 file_existsO 函 数 测 试 一 个 文件 的 存在 性 。 这 个 函数 需 
要 表示 一 个 文件 的 绝对 路 径 或 者 相对 路 径 的 一 个 字符 串 ， 访 文件 可 能 存 
在 也 可 能 不 存在 。 如 果 找 到 该 文 件 ，file_exists() 函 数 返 回 true; FUE 
1 [Hl false. 
if (file exists{'test.txt')} d 


echo “The file exists!’; 


} 

REC AB EH, Bæ, WRA REE KEP OC SAHK, 
并 且 真 的 需要 知道 ， 那 该 怎么 办 呢 ? 
13.2.2 文件 还 是 目录 

我 们 可 以 使 用 is_file() 函 数 来 确定 所 要 测试 的 实体 是 一 个 文件 ， 或 者 
是 一 个 目录 。is_file(0) 函 数 宝 要 文件 的 路 径 并 且 会 返回 一 个 布尔 值 。 
It (IS EEL Pest. Eke) f 

echo “test.txt is a file!"; 


i 
FA, BATA) Fe A or AMEMA ce Be Ae, FY DAE AY 


is_dirO RRR MAI IX— A. is_dir() fi H Kh EA EVENS BF H eE — 
个 布尔 值 。 


if (is_dir('/tmp')) { 
echo "/tmp is a directory"; 


} 


当 我 们 知道 一 个 文件 或 目录 存在 后 ， 可 能 再 要 测试 其 是 合 允 许 访 
问 。 我 们 将 在 下 一 区 学 习 这 一 点。 


13.2.3 ”检查 一 个 文件 的 状态 


当知 着 一 个 特定 的 实体 存在 ， 并 且 它 古 我 们 所 期 望 的 一 个 目录 或 者 
一 个 文件 ， 我 们 将 十 要 知道 可 以 对 它 做 什么 。 通 第 ， 你 可 能 希望 读 取 、 
写 入 或 执行 一 个 文件 。PHP 可 以 帮助 你 确定 是 任 可 以 执行 这 些 操 作 。 


is_readable() 函 数 告诉 你 是 人 否 可 以 读 取 一 个 文件 。 在 UNIX 系 统 上 ， 
我 们 或 许 能 够 看 到 一 个 文件 ， 但 是 仍然 禁止 读 取 其 内 容 ， 因 为 需要 用 户 
诗 可 。is_readable() 阔 数 搁 受 一 个 字符 串 作 为 文件 路 人 径 ， 并 且 返 回 一 个 
布尔 值 。 


if (is_readable('test.txt'}) { 
echo "test.txt is readable"; 


} 


is_writable0) 孙 数 告 诉 你 是 舍 拥 有 写 入 一 个 文件 的 许可 。 和 
is_readable() 一 样 ，is_writable() 疯 数 需 要 一 个 文件 路 入 并 且 返 回 一 个 布 
尔 值 。 
if (is_writable('test.txt')) { 


echo "test.txt is writable"; 


} 


is_executable() K AN eo UE Ee AT ce Fo AY WA A RE CEN FY Bae 
展 名 ， 在 你 的 特定 平台 上 执行 它 。 这 个 函数 接受 文件 的 路 径 并 且 返 回 一 
个 布尔 值 。 


if (is_executable{'test.txt')) { 
echo ‘test.txt 18 executable"; 


} 


和 许可 相关 的 信息 并 不 是 我 们 需要 知道 的 关于 一 个 文件 的 全 部 信 
居 。 下 一 节 将 展示 如 何 确 定 文件 的 大 小 。 
13.2.4 ”使 用 filesizeO 确 定 文件 的 大 小 

给 定 了 文件 的 路 径 ，filesize0 函 数 答 试 确定 并 返回 文件 的 大 小 《以 
字 节 为 单位 ) 。 如 果 过 到 问题 ， 它 返回 false， 示 例如 下 。 
echo "The size of test.txt is ".filesize{(' test.txt'}; 

当 我 们 想 要 把 一 个 文件 作为 一 封 Email 的 附件 或 者 想 要 把 文件 传送 
给 用 户 的 时 候 ， 知 道具 体 文件 的 大 小 是 很 重要 的 ， 知 道 文件 的 大 小 ， 在 
Email 的 情况 下 以 便 正 确 地 创建 标 头 或 者 在 流传 大 的 情况 下 知道 何 时 个 
止 回 用 户 及 送 宇和 节 。 对 更 通用 的 情况 而 言 ， 我 们 可 能 想 获 取 文 件 的 大 
小 ， 以 便 可 以 在 用 尸 试 图 从 你 的 站 点 下 载 超 大 容量 的 应 用 程序 或 者 局 清 
晰 大 的 照片 之 前 ， 将 文件 大 小 显示 给 他 们 。 
13.25 ”获取 有 关 一 个 文件 的 日 期 信息 


有 时 候 ， 我 们 需要 知道 一 个 文件 最 后 一 次 写 入 或 访问 的 时 间 。PHP 
提供 的 几 个 函数 可 以 提供 这 些 信息 。 


我 们 可 以 使 用 featime0) 函 数 来 获取 一 个 文件 的 最 后 访问 时 间 。 这 


AS PRIA ma BECP RIE FAK lela DA SCAR BEG eA 
1, Bae ae SAE. MAA HANE OS AROR EA H HAER 
IRS, 也 就 是 说 ， 从 1970 年 1 月 1 日 起 的 秒 数 。 在 我 们 的 例子 中 ， 
使 用 date(O) 函 数 来 把 这 个 值 转换 为 人 们 可 以 识别 的 形式 ， 示 例如 下 。 


$atime = fileatimet'"test.txt")}; 
echo "test.txt was last accessed on ".date({("D d M Y gii A", $atime), 
{£ Sample output: test.txt was last changed on Sat 26 Apr 2008 12:52 PM 


BAT BY E H filemtime() ci ALK SRP PEI A. KER 
数 需要 一 个 文件 路 径 ， 并 且 以 UNIX 时 间 惟 格式 返回 日 期 。 要 修改 一 个 
文件 ， 意 味 看 以 采种 方式 修改 其 内 容 ， 示 例如 下 。 


$mtime = Filemtime("test.txt"}; 
echo “test .txt was last modified on “.date("D d M Y g:i A", $mtime); 
{í Sample output: test.txt was last modified on Wed 18 Jan 2012 7:11 PM 


PHP 也 人 允许 我 们 使 用 filectime() eA BM i— Ph CE RAAT. Æ 
UNIX 系统 上 ， 当 一 个 文件 的 内 容 被 修改 或 者 当 其 许可 权限 或 所 有 者 友 
生变 化 的 时 候 ， 束 设置 修改 日 期 。 在 其 他 的 平台 上 ， 人 lectime() 函 数 返 
回 文 件 的 创建 日 期 ， 示 例如 下 。 


mtime = filemtime("test.txt"}; 
echo "test.txt was last modified on ".date("D d M Y g:i A", $Smtime}; 
// Sample output: test.txt was last modified on Wed 18 Jan 2012 7:11 PM 


13.2.6 ”编写 一 个 执行 多 文件 测试 的 函数 


程序 清单 13.7 创 建 了 一 个 函数 ， 它 把 我 们 了 刚刚 介绍 的 一 些 和 文件 相 
天 的 函数 引入 到 一 个 脚本 中 。 





程序 清单 13.7 输出 多 个 文件 测试 的 结果 的 一 个 函数 


ls <?php 

2: function outputFileTestInfo($f) { 

g: if (!file_exists($f)) 1 

4: echo "<p>$f does not exist</p>"; 

5i return; 

&: } 

Ta echo "<p>$f is ".(is_file($f) ? "" : "not ")."a file</p>"; 

8: echo “so>St ps "afis dr T 7E = "net "kha directory<for": 

9: echo "<p>$f is ".(is_readable($f} ? "": "not "}."readable</p>"; 

10: echo "<p>$f is ".{is writable($f}) ? "": "not ")."writable</p>'"; 

113 echo "<p>$f is ".({is_executable($f) 7? "": "not ")."executable</p>"; 

12: echo "<p>$f is ".{filesize($f)}." bytes</p>"; 

133 echo "<p>$f was accessed on ".date( "D d M Y g:i A",fileatime($f))."</p>"; 
14: echo "<p>$f was modified on ".date( "D d M Y g:i A",filemtime($f))."</p>"; 
oe echo "<p>$f was changed on ".date( "D d M Y g:i A",filectime($f) )."</p>"; 
TOS } 

17: $file = "test.txt"; 


18: outputFileTestInfo($file); 


Tar > 


如 果 这 
下 ， 并 且 通 


段 代码 作为 filetests.php 保存 到 Web 服务 需 的 文档 根 目录 


过 Web 浏览 带 来 运行 ， 输 出 将 会 如 图 13-2 所 示 。 





[=] localhost/13/filetests.php 


& © | © http://localhost/13/filetests.ohp a, 
test.txt is a file 

test.txt is not a directory 

test.txt is readable 

test.txt is writable 

test.txt is mot executable 

test txt is 34 bytes 

test.txt was accessed on Sat 26 Apr 2008 12:52 PM 

test.txt was modthed on Wed 18 Jan 2012 7:11 PM 


testixt was changed on Sat 26 Apr 2008 12:52 PM 


(413-2  filetests.php HJ 447 H 


VE Rsk, TEREIT A EE MUTA ES R18 A = CBR EST EAP AY 
方式 。 让 我 们 更 详细 地 看 看 这 样 一 个 测试 ， 如 程序 清早 13.7 的 第 7 行 所 
不 。 


echo "<p>$f is ".{is file($f) ? "" : "not ")."a file</p>"; 


我 们 使 用 is_fileQ ee BF A= JcPR ERT Ze AAT. BNR ER E 
tue, Heskll—PaA yee, AML, Roklel eer not”. SRE 
A PRIE Ze BRE FS 0B EE AN YAP AB. KR RT A) BY De 
展 变 得 更 清楚 些 ， 如 下 所 示 。 


$is it = is Tile({$f) 7 "" : "not "; 
echo "<p>" .$f." is ".$1s it." a file</p>"; 


当然 ， 我 们 可 以 设置 用 一 条 if 语 句 使 其 变 得 更 清楚 ， 不 过 想象 一 
和 下， 如果 使 用 如 下 的 形式 ， 函 数 将 会 变 得 多 长 。 


if {is file($f)) { 
echo "<p>$f is a file</p>'"; 
} else { 
echo "<p>$f is not a file</p>"; 


} 


由 于 这 3 种 方法 的 结束 是 一 样 的 ， 选 择 哪 种 方法 由 个 人 俩 好 而 定 。 


13.3 创建 并 删除 文件 


如 果 一 个 文件 还 不 存在 ， 我 们 可 以 使 用 touch() 函 数 来 创建 它 。 给 定 
一 个 表示 文件 路 径 的 字符 串 ，touch() 函 数 试图 使 用 该 名 字 创 建 一 个 空 文 
件 。 如 果 这 个 文件 已 经 存在 ， 其 内 容 不 会 被 改变 ， 但 是 修改 日 期 将 会 
新 ， 以 反映 出 该 函数 的 执行 时 间 。 


touch('myfile.txt'}; 

我 们 可 以 使 用 unlink0 函 数 来 删除 一 个 已 有 的 文件 ， 束 像 touch0 函 数 
一 样 ，unlink0 也 接 党 一 个 文件 路 径 。 
ULEnmRE mLLeu tt F; 

在 UNIX 系 统 上 创建 、 删 除 、 读 取 、 写 入 和 修改 文件 的 所 有 水 数 都 
着 要 设置 正确 的 文件 或 目录 许可 。 


13.4 打开 一 个 文件 供与 入、 侠 取 或 述 加 


在 我 们 开始 操作 一 个 文件 之 前 ， 必 须 先 打开 它 以 便 读 取 或 写 入 ， 或 
者 这 两 种 任务 都 执行 。PHP 提供 了 fopen0 函 数 来 做 到 这 一 点 ， 并 且 这 
个 函数 需要 一 个 包含 了 文件 路 径 的 字符 串 ， 后 面 跟着 一 个 字符 串 ， 其 中 
包含 了 文件 的 打开 模式 。 最 常见 的 模式 是 读 取 GD. BA Cw) MAD 
Ca) 。 


fopen() 疯 数 返 回 一 个 文件 源 ， 我 们 和 后 可 以 使 用 它 来 操作 被 打开 的 
文件 。 要 打开 一 个 文件 以 供 读 取 ， 我 们 使 用 如 下 的 方式 。 


$fp = fopen("test.txt", "r"}; 


使 用 如 下 方式 来 打开 一 个 文件 以 便 写 入 。 
$fp = fopen("test.txt", "w"}; 

要 打开 一 个 文件 以 便 添 加 内 容 ， 即 在 文件 末尾 添加 数据 ， 我 们 使 
用 ， 如 下 方式 。 
Sfp = fopen("test.txt", "a"}; 

Qn 52 A VE JAI AS EFT IP CHE, fopen0 AROR [Flfalse. KHE, 


在 es 数 的 返回 值 ， 是 个 不 错 的 方法 。 我 们 可 以 使 用 如 
下 一 条 让 语句 来 做 到 这 一 点。 


if ($fp = fopen("test.txt", "w")} { 


// do something with the $fp resource 


} 
或 者 ， 如 朱 一 个 基本 文件 无 法 打开 ， 我 们 可 以 使 用 一 个 馆 辑 操作 符 


来 结束 执行 ， 示 例如 下 。 


($fp = fopen("test.txt", "w")) or die("CGouldn't open file, sorry"}; 
tO Rfopen() eA BE Eltrue, KARR PUA ASAT, Elldie() 
KHA TARR ALS AA ES AAA SSP RIAN) 不 会 执行 。 否 则 ， 
or 操作 符 的 右 侧 表 达 式 将 会 解析 ， 即 调用 dieO 函 数 。 


假设 这 一 切 都 正常 ， 并 用 我 们 继续 使 用 打开 的 文件 ， 当 我 们 完成 的 
时 候 应 该 记 得 关闭 它 。 我 们 可 以 通过 调用 fclose0O 函 数 来 做 到 这 一 点 ， 这 
需要 从 一 次 成 功 的 fopenO 调 用 所 返回 的 文件 源 作为 参数 。 


fclose($fp}; 


曾经 可 用 的 文件 源 Sfp) 现在 对 你 已 经 不 可 用 了 。 


13.5 ”该 取 文件 


PHP 提 供 了 几 个 函数 来 从 文件 谈 取 数据 。 pei Ba (HE FS ER ANT ES FX 
ii 


13.5.1 ”使 用 fgets() 和 feof() 从 一 个 文件 读 取 行 


当 我 们 打开 一 个 文件 以 供 读 取 的 时 候 ， 你 可 能 想 一 行 一 行 地 访问 
它 。 要 从 一 个 打开 的 文件 谈 取 行 ， 我 们 可 以 使 用 fgets0 函 数 ， 这 需要 从 
fopenO 返 回 的 文件 源 作 为 其 参数 。 我 们 还 必须 传递 一 个 整数 作为 fgets) 
的 第 二 个 参数 ， 它 指明 了 如 采 函 数 没 有 遇 到 一 个 行 末 或 文件 末尾 而 应 该 
读 取 的 字 市 数 。fgets() 函 数 读 取 文件 ， 和 直到 它 过 到 一 个 换行 字符 
(Aw ， 或 者 达到 长 度 参 数 中 指定 的 字 节 数 ， 或 者 是 到 达 文 件 的 末 
Fe, METIS, AIM F 
$line = fgets($fp, 1024}; // where $fp is the file resource returned by fopen() 

尽管 可 以 使 用 fgetsO) 来 读 取 行 ， 但 是 我 们 需要 菜 种 方法 来 告知 何 时 
到 达 文 件 的 末尾 。feofO 函 数 可 以 做 到 这 一 点 ， 当 到 达 了 文件 的 末尾 的 


时 候 ， 该 函数 返回 true， 否 则 返回 false。feof0) 函 数 需要 一 个 文件 源 作为 
参数 ， 不 例如 下 。 


feof($fo); // where $fp is the file resource returned by fopen(} 


现在 ， 我 们 有 足够 的 信息 来 一 行 一 行 地 谈 取 一 个 文件 ， 如 程序 清单 
13.8 所 示 。 


程序 清单 13.8 打开 一 个 文件 并 一 行 一 行 地 读 取 


: <?php 

: $filename = "test.txt"; 

: $fp = fopen($filename, "r"} or die("Couldn't open $filename"); 
: while (!feof($fp)) { 

$line = fgets($fp, 1024); 

echo $line. "<br/>"; 


co ~ 中 加 上 有一 


1 

把 上 述 代 码 作为 readlines.php 保 存 到 Web 服 务 器 的 文档 根 目录 下 ， 
并 且 在 Web 浏 览 器 中 运行 它 ， 输 出 将 会 如 图 13-3 所 示 (你 的 示例 文本 文 
件 的 内 容 可 能 会 有 所 不 同 ) 。 





B localhost/13/readlines.php 
f= C | © http://localhost/13/readlines.ohp a, 


Hello world! 
And another thing... 


图 13-3 ”readlines.php 的 输出 


我 们 在 第 3 行 调用 fopen0O 函 数 ， 使 用 想 要 旋 取 的 文件 的 名 字 作 为 参 
数 。 如 果 没 有 读 取 文件 ， 我 们 使 用 or die() 结 构 来 确保 结束 脚本 的 执行 。 
如 果 文 件 不 存在 或 文件 的 许可 不 允许 访问 这 个 文件 ， 通 党 会 友 生 这 种 情 
况 。 


文件 内 容 的 实际 读 取 发 生 在 第 4 行 的 while 语句 。while 语 句 的 测试 
表达 式 每 次 迭代 都 调用 feof0) 疯 数 ， 当 它 返 回 true 的 时 候 循环 结束 。 换 
句 话说 ， 循 环 持续 直到 到 达 文 件 末 尾 。 在 代码 块 内 部 ， 我 们 在 第 5 行使 
用 fgets() 从 文件 提取 一 行 (或 者 1024 字 节 ， 不 管 这 一 行 够 不 够 1024 F 
W) 。 我 们 把 结果 赋 给 $line 并 有 旦 在 第 6 行将 其 显示 到 浏览 器 ， 为 了 增强 
可 读 性 ， 在 最 后 附加 一 个 <br/> 标 记 。 


13.5.2 ”使 用 fread0) 孙 数 从 文件 伟 取 任意 数量 的 数据 


我 们 可 以 选择 按照 任意 定义 的 大 小 来 读 取 一 个 文件 ， 而 不 是 按照 行 
来 读 取 文本 。fread() 函 数 接受 一 个 文件 产 以 及 和 想 要 读 取 的 字 市 数 作 为 参 
数 。fread() 了 水 数 返 回 我 们 所 请 求 的 那么 多 的 数据 ， 除 非 到 达 文 件 的 末 
尾 。 


$chunk = fread($fp, 16}; 


程序 清单 13.9 修改 了 前 面 的 例子 ， 以 便 以 8 字 节 的 数据 块 来 读 取 数 
据 ， 而 不 十 按照 行 来 读 取 。 


如 果 把 这 段 代 码 作 为 readlines2.php 文 件 保 存 到 Web 服 务 豆 的 文档 根 
目录 下 ， 并 有 旦 通过 Web 浏 贤 右 来 运行 它 ， 将 会 看 到 如 图 13-4 所 示 的 结 
FR 


O N nab wah — 


=o 
[=] localhost/13/readlines2.ph; 


€ C | © http;/localhost/13/readlines2.php x, 
Hello wo 
rid! And 


another 


thing.. 


图 13-4 ”readlines2.php 的 输出 


程序 清单 13.9 ”使 用 fread0 谍 取 文 件 


: <?php 
: $filename = "test.txt"; 


Sfp = fopen($filename, "r"} or die("Couldn't open $filename"); 
while (!feof($fp)) { 

$chunk = fread($fp, 8); 

echo $chunk. "<br/>"; 


FS fread() Pl BIC VEU RE MMP SPER Be, (Ee, E 


不 允许 你 决定 从 哪个 位 置 开始 获取 。 我 们 可 以 使 用 fseek() 函 数 来 手动 地 


WEE o 


fseek() 函 数 允 许 我 们 在 一 个 文件 中 修改 当前 位 置 。 它 需要 一 个 文件 


源 以 及 一 个 整数 作为 参数 ， 这 个 整数 表示 从 文件 的 开始 跳 过 的 偶 移 量 
(学 市 数 ) 。 


fseek{$fp, 64); 


程序 清单 13.10 使 用 fseek() 和 freadO) 来 把 一 个 文件 后 半 部 分 输出 到 浏 


程序 清单 13.10 ”使 用 fseek() 来 移动 文件 

Pe “SY Drip 

2: $filename = “test.txt"; 
3: $fp = fopen($filename, "r") or die{"Couldn't open $filename"); 
4: $fsize = filesize($filename) ; 
5: $halfway = (int)($fsize / 2); 
6: echo "Halfway point: ".$halfway." <br/>\n"; 
7: fseek($fp, Shalfway) ; 
8: $chunk = fread($fp, ($fsize - $halfway)); 
9: echo $chunk; 

10: ?> 


W RHE ERIE Areadseek.phpte4¢ 2] Weblk 3 4s HY) SC RS AR H Se 
下 ， 并 且 通 过 Web 浏 览 器 来 运行 它 ， 将 会 看 到 如 图 13-5 所 示 的 结果 。 


localhost/13/readseek.php 
€ @ | © http://localhost/13/readseek.php A, 


Halfway point: 17 
another thing... 


图 13-5 readseek.php 的 输出 


我 们 在 第 5 行 通 过 把 flesize0 的 返回 值 除 以 2， 计 算出 文件 的 一 半 的 
位 置 。 在 第 7 行 ， 把 这 个 位 置 用 作 fseekO 函 数 的 第 2 个 参数 ， 跳 到 这 个 文 
本 文件 一 半 有 的 位 置 。 最 后 ， 在 第 8 行 调用 fread0 〇 函数 来 提取 文件 的 为 一 
半 ， 并 且 将 结 来 输出 到 显示 此 。 


13.5.3 ”使 用 fgetc() 从 文件 读 取 字符 


fgetc()RI ALA fgets() 函 数 很 相似 ， 只 不 过 每 次 调用 它 的 时 候 只 从 文 
件 返 回 一 个 字符 。 由 于 一 个 字符 总 是 一 个 字 节 的 大 小 ，fgetc0 不 需要 长 
上 度 参数 。 我 们 只 需 传 递 给 它 一 个 文件 产 ， 示 例如 下 。 


$char = fgetc({$tp}; 


程序 清单 13.11 创建 了 一 个 循环 ， 它 每 次 从 test.txt 文件 读 出 一 个 字 
符 ， 把 每 个 字符 作为 单独 一 行 输出 到 浏览 右 中 。 


程序 清单 13.11 使 用 fgetc() 来 移动 文件 


2: $filename = “test.txt"; 
3: $fp = fopen($filename, "r"} or die("Couldn't open $filename"); 
4: while (!feof($fp)) { 
5: $char = fgetc($fp); 
6: echo $char."<br/>"; 
T: 9} 
B: te 
如 果 把 上 述 代 码 作为 readchars.php 文 件 保 存 到 Web 服 务 器 的 文档 根 
目录 下 ， 并 且 通 过 Web 浏 宽 如 来 运行 它 ， 将 会 看 到 如 图 13-6 所 示 的 结 


AR 





localhost/13/readchars.php 
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图 13-6 readchars.php 的 输出 


13.5.4 用 file_get_contents0 读 取 文 件 内 容 

现在 已 经 掌握 了 使 用 fopen() 来 创建 一 个 文件 资源 并 且 对 该 资源 进行 
一 项 或 多 项 操作 ， 可 以 通过 使 用 file_get_contents() 将 整个 文件 读 入 到 一 
个 字符 串 中 ， 从 而 来 做 各 种 操作 。 例 如 ， 如 下 的 单行 代码 ， 将 名 为 
test.txt 的 文件 的 内 容 读 入 到 一 个 名 为 $contents 的 变量 中 。 


$contents = file get contents("test.txt"}; 


这 个 函数 还 提供 了 一 些 其 他 的 可 能 性 ， 下 面 给 出 了 调用 
file_get_contents() 时 候 可 用 的 选项 。 


file get contents ($filename [Suse include path [, $context [, $offset [, 
$maxlen ]]]1); 


这 些 选 项 插 述 如 下 。 


use_include_path 一 一 这 是 一 个 布尔 值 ， 它 表明 函数 是 耕 应 该 搜索 
文件 名 的 完整 include_path。 

这 是 stream_context_create() 创 建 的 一 个 上 下 文 资源 ， 如 
果 不 和 需要 的 话 ， 使 用 NULL。 

谈 取 从 哪里 开始 ， 例 如 ，5 表 示 文 件 的 第 5 个 字符 。 注 
意 ， 可 以 对 非 本 地 文件 〈 例 如 URL ) 使 用 file_get_contents0， 在 这 
种 情况 下 ， 不 能 使 用 offset。 

BEAT IRATE, WRAL. BRUTE Pi 
取 所 有 的 数据 。 





e context 


e offset 





e maxlen 





如 果 前 面 函数 的 精细 上 度 控 制 在 你 的 脚本 中 不 是 必须 的 ， 那 么 ， 
file_get_contents()t8 E EI E18 4 ZEA ALR AY o 


13.6 SASSI MAINE 


写 入 文件 与 同文 件 添加 内 容 的 过 程 是 相同 的 ， 不 同 之 处 仅 在 于 它们 
调用 fopen0) 函 数 的 模式 。 当 我 们 写 入 一 个 文件 的 时 候 ， 调 用 fopen0) 函 数 
的 时 候 可 以 使 用 模式 参数 “w”， 丰 例如 下 。 
$fp = fopen("test.txt", "w"}; 

所 有 后 续 的 写 入 部 发 生 在 文件 的 开 尖 处 。 如 果 这 个 文件 还 不 存在 ， 
束 创 建 它 。 如 采 这 个 文件 已 经 存在 ， 之 前 的 所 有 和 内容 都 销毁 并 且 由 你 与 
入 的 数据 百代 。 

当 我 们 添加 到 一 个 文件 时 ， 在 fopenO 调 用 中 使 用 模式 参数 “a”， 示 
例如 下 。 
$fp = fopen("test.txt", "a"}; 

AA a Ze S ASCP A ASC SI BU A AN RRE. URA 
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13.6.1 ”使 用 fwrite() 或 fputs0 写 入 文件 
fwrite() 逊 数 接受 一 个 文件 源 和 一 个 字符 串 ， 然 后 把 字符 串 写 入 到 文 
件 中 。fputsO0 函 数 也 以 相同 的 方式 工作 ， 示 例如 下 。 


fwrite($fp, "hello world"); 
fputs($fp, "hello world"); 


写 入 到 文件 相当 人 简单。 程序 清单 13.12 使 用 fwrite0 写 入 到 文件 。 然 
后 ， 我 们 使 用 fputsO 把 另 一 个 字符 串 添 加 到 同一 个 文件 。 


程序 清单 13.12 ” 写 入 文件 和 向 文件 添加 内 容 


ta =7DNp 

2: $filename = "test.txt"; 

3: echo "<p>Writing to ".$filename." ... </p>"; 
4: $fp = fopen($filename, "w") or die{"Couldn't open $filename"); 
5: fwrite($fp, "Hello world\n"); 

6: fclose($fp); 

7: echo "<p>Appending to ".$filename." ...</p>"; 
8: $fp = fopen($filename, "“a") or die{"Couldn't open $filename"); 
9: fputs($fo, "And another thing\n"); 

10: fclose($fp) ; 

1 ie > 


BAT Webi ArT ANAR, Bee FT HE UM FE 


Writing to test.txt ... 
Appending to test,.txt ... 


如 条 打开 test.txt 文 件 或 者 使 用 readlines.php 来 谈 取 其 内 容 ， 你 将 会 用 
现 文件 现在 包含 如 下 内 容 。 


Hello world 
And another thing 


13.6.2 ”使 用 多 e put_contents0) 与 文件 内 容 


和 前 面 介 绍 的 fie_get_contentsO 函 数 很 相似 ，fie_put_contentsO 函 数 
以 一 种 更 为 流 线 化 的 方式 来 进行 文件 操作 。 特 别 的 ，file_put_contents() 
直接 模拟 了 依次 调用 fopen()、fwriteO 和 fclose()， 来 将 数据 写 入 文件 。 


下 面 展 示 了 调用 file_put_contents0O) 的 时 候 允 许 使 用 的 参数 。 


file put contents ($filename, $data [, $flags [, $cantext]]}; 
这 些 额外 的 选项 说 明 如 下 。 


包 舍 了 要 与 的 数据 的 一 个 字符 串 或 数组 。 


e data 





一 个 或 多 个 FILE_USE_INCLUDE_PATH (是 否 在 
include_path 中 查找 目标 文件 ) 、FILE_APPEND (如 果 文 件 已 经 存 
在 的 话 ， 是 全 将 数据 洪 加 到 一 个 文件 后 面 ) 和 LOCK_EX ERE 
写 入 目标 文件 时 获取 一 个 专用 的 锁 ) ， 这 些 选项 用 | 操作 符 组 合 起 
K 


e context 


e flags 





stream_context_createO 创 建 的 一 个 环境 资源 ， 如 采 不 需 
要 的 十 ， 使 用 NULL。 





如 果 使 用 fie_put_contents0) 重 新 编写 程序 清单 13.12， 它 看 上 去 如 程 
序 清单 13.13 所 示 。 


程序 清单 13.13” 写 入 并 诬 加 到 一 个 文件 


1: <?php 

2: $filename = "test.txt"; 

3: echo "<p>Writing to ".$filename." ... </p>"; 

4: file put_contents ($filename, "Hello world\n"); 

5: echo "<p>Appending to ".$filename." ...</p>"; 

6: file put_contents ($filename, "And another thing\n", FILE APPEND) ; 
fi > 


13.6.3 ”使 用 flock0 锁 定 文 件 


如 果 只 有 一 个 用 户 访 问 脚本 的 话 ， 刚 刚 学 习 的 读 取 和 补充 文件 的 方 
法 会 工作 得 很 好 。 然 而 ， 在 现实 使 用 中 ， 我 们 期 望 更 多 的 用 户 可 以 访问 
站 点 以 及 其 中 的 脚本 ， 所 以 或 多 或 少 会 有 同时 的 访问 。 假 设 有 两 个 用 户 
要 同时 执行 写 入 到 一 个 文件 的 脚本 ， 这 个 文件 会 很 快 面目 全 非 。 


PHP 提 供 了 flock() 函 数 来 避免 这 种 问题 。flock() 孙 数 锁定 一 个 文 
件 ， 以 鸭 各 其 他 的 进程 ， 在 当前 进程 使 用 这 个 文件 的 时 低 ， 不 要 再 写 入 
或 该 取 这 个 文件 。flockO 函 数 需 要 一 个 来 目 打开 的 文件 的 有 效 文件 源 以 


太一 个 整数 ， 这 个 整数 表示 你 想 要 设置 的 锁定 的 种 类 。PHP 为 我 们 可 能 
再 要 的 每 个 整数 都 捉 供 了 预定 义 的 利 量 。 雪 13-1 列 出 了 我 们 可 以 应 用 于 
一 个 文件 的 3 种 锁定 。 


表 13-1 ”flock0 函 数 的 整数 参数 


类 型 说 明 





允许 其 他 进程 读 取 文件 但 是 阻止 写 入 在 读 取 一 个 文件 的 时 
候 使 用 ) 


阻止 其 他 进程 读 取 文 件 或 者 写 入 文件 “在 写 入 一 个 文件 时 使 
FA) 


释放 一 个 共 圣 锁定 或 独占 锁定 





我 们 应 该 在 调用 fopen() 之 后 直接 调用 flock()， 然 后 ， 在 关闭 一 个 文 
件 之 六 再 次 调用 它 来 释放 锁定 。 如 果 锁 定 没 有 释放 ， 将 不 能 够 谈 取 或 写 
入 文件 。 下 面 是 这 个 事件 序列 的 一 个 例子 。 


$fp = fopen("test.txt", "a") or die("Couldn't open file."}; 
flock($fp, LOCK_EX); // create exclusive lock 

i write to the file 

Flock($fp, LOCK UN); // release the lock 

felose(Sfp); 





你 知道 吗 ? 要 了 解 关 于 文件 锁定 的 更 多 信息 ， 请 参见 位 于 http://www.php. net/flocks 的 PHP 


手册 中 的 flock0) 函 数 条 目 。 | 


13.7 ”使 用 目录 


既然 可 以 创建 、 测 试 和 写 入 文件 ， 那 么 让 我 们 把 注音 力 转 问 目 录 。 
PHP 近 供 了 很 多 用 来 使 用 目录 的 函数 。 让 我 们 看 一 下 如 何 创 建 目 录 、 删 
际 目 隶 和 从 中 读 取 内 容 。 


13.7.1 ”使 用 mkdir0 创 建 目录 


mkdir0 函 数 使 得 我 们 能 够 创建 一 个 目录 。mkdir0 函 数 需要 一 个 字符 
串 和 一 个 八进制 整数 ， 字 符 串 表示 到 我 们 想 要 创建 的 目录 的 路 径 ， 整 数 
表示 我 们 想 要 为 目录 设置 的 模式 。 别 筷 了 ， 我 们 使 用 0 开头 来 指定 一 个 
八进制 数 《〈 以 8 为 基数 ) ， 例 如 0777 或 0400。 


模式 参数 只 在 UNIX 系 统 下 有 效 。 这 个 模式 应 该 由 从 0 到 7 之 间 的 3 个 
数字 组 成 ， 它 们 分 别 表 示 对 目录 所 有 者、 组 和 任何 人 的 许可 权限 。 如 果 
成 功 地 创建 了 目录 ，mkdir0 函 数 返 回 true， 合 则 返回 false。 如 采 mkdir0) 
失败 ， 通 剃 是 因为 包含 目录 不 允许 市 有 脚本 的 用 户 ID 的 进程 写 入 。 


如 果 我 们 不 知道 如 何 正 确 地 设置 UNIX 目 录 许可 ， 如 下 例子 中 的 一 
个 应 该 符合 你 的 需求 。 除 非 真 的 需要 将 自己 的 目录 设置 为 完全 可 写 的 ， 
否则 你 可 能 应 该 使 用 0755， 它 使 得 目录 完全 可 读 但 是 无 法 向 其 中 写 入 ， 
除了 目录 的 所 有 者 之 外 ， 


mkdir¢"testdir’, 0777); // global read/write/execute permissions 
mkdir("testdir’, 0755}; // world/group: read/execute; owner: read/write/execute 


13.7.2 ”使 用 rmdir0 删 除 一 个 目录 


如 来 运行 脚本 的 进程 有 权利 这 么 做 ， 并 且 目 录 为 空 的 话 ，rmdir() 函 
数 允 许 我 们 从 文件 系统 删除 一 个 目录 。rmdir0) 函 数 只 需要 一 个 字符 串 作 
为 参数 ， 它 表示 到 第 要 删除 的 目录 的 路 任 。 


rmdir("testdir'); 


13.7.3 ”使 用 opendir() 打 开 一 个 目录 以 供 证 取 


在 阅读 一 个 目录 的 内 容 之 前 ， 必 须 首 先 获得 一 个 目录 源 。 我 们 可 
以 使 用 opendirO 函 数 来 做 到 这 一 点 。opendirO 函 数 需 要 一 个 字符 串 ， 它 
表示 要 打开 的 目录 的 路 径 ，opendir() 函 数 返回 一 个 目录 人 句柄， 际 非 这 个 
目录 不 存在 或 者 不 可 读 。 如 末 目 录 人 不 存在 或 者 个 可 读 ， 它 返回 false， 如 
下 所 示 。 


$dh = opendir{"testdir'); 
在 这 个 例子 中 ，$dh 束 是 打开 目录 的 目录 句柄 。 
13.7.4 使 用 readdir0 从 一 个 目录 谍 取 内 容 


束 像 使 用 fgets0 函 数 从 一 个 文件 中 读 取 一 行 一 样 ， 我 们 可 以 使 用 
readdir() 从 一 个 目录 读 取 一 个 文件 或 目录 名 。readdir() 疯 数 需 要 一 个 目录 
句柄 并 且 返 回 包含 项 目 名 的 一 个 字符 串 。 如 采 到 达 了 目录 的 末尾 ， 
readdir() 人 返回 false。 注 意 ，readdir() 只 是 返回 其 项 目的 名 字 ， 而 不 是 完整 
的 路 径 。 程 序 清单 13.14 显 示 了 一 个 目录 的 内 容 。 


程序 清单 13.14 “使 用 readdir0 列 出 一 个 目录 的 内 容 


<?php 
$dirname = "."; 
: $dh = opendir($dirname) or die("Couldn't open directory"); 


while (!(($file = readdir($dh)) === false }) ) { 
if (is_dir("$dirname/$file")) { 
echo "(D) "; 


ON Oa hh aN — 


ce) 


echo $file."<br/>"; 


} 
11: closedir($dh) ; 
tex! t= 


HE ERINE Areaddir.php tr tt #!] Webs ó a8 HCE Ae RSF 
日 通 过 Web 浏 览 器 来 运行 它 ， 其 输出 如 图 13-7 所 示 。 


[=] localhost/13/readdir.php 


各 CG | © htto://localnost/13/readdir.php a, 


(D). 
(D) .. 
exec Is php 
filetests php 
getbiary php 
mchle 1] txt 
mchle? txt 
mchle3 txt 
loopy mclude php 
mymchide php 
mymclude2 php 
popen column php 
popen who. php 
readchars php 
readdir php 
readies php 
readines. php 
readseek php 
returnvalue php 
test. txt 
test inchude php 

| test include2 php 


mW 


test_popen php 
test_returnvalue php 


wahile php S 


图 13-7 readdir.php Hy 447 14 


我 们 在 第 3 行使 用 opendirO 函 数 打开 目录 以 便 访 取 ， 并 且 从 第 5 行 开 
始 ， 使 用 一 条 while 语 句 来 过 历 其 中 的 每 个 元 素 。 我 们 调用 readdirO 语 名 
作为 while 语 句 的 测试 表达 式 的 一 部 分 ， 并 且 将 其 结果 赋 给 $file 变 量 。 


在 while 语句 体 的 内 部 ， 我 们 使 用 $dirmame 变量 和 $file 变量 一 起 来 
创建 一 个 完整 的 文件 路 径 ， 然 后 ， 在 第 6 行 负 斌 这 个 路 径 。 如 条 这 个 路 


径 表 示 一 个 目录 ， 我 们 让 第 7 行 在 浏览 器 上 显示 (D) 。 最 后 ， 在 第 9 行 
显示 文件 名 或 目录 名 。 


程序 清单 13.14 在 while 语 句 的 训 试 中 使 用 了 一 个 讶 慎 的 结构 。 大 多 
数 PHP 程 序 员 《包括 我 目 己 在 内 ) 者 愿意 使 用 如 下 所 示 的 代 但 。 


while ($file = readdir($dh)) { 
echo $file."<br/>": 


I 


在 这 个 例子 中 ， 测 试 readdir0) 的 返回 值 ， 并 有 日 ， 由 于 除 “0” 以 外 的 任 
何 字 人 符 串 都 求 值 为 tue， 这 里 不 会 有 什么 问题 。 然 而 ， 假 设 一 个 目录 包 
含 4 个 文件 : 0、1、2 和 3。 在 我 日 己 的 系统 上 ， 前 面 的 代码 输 出 如 下 。 


当 循 环 到 达 名 为 0 的 文件 的 时 候 ，readdir0 返 回 的 字符 串 求 值 为 
false， 这 会 叶 色 循环 结束 。 程 序 清单 13.14 中 的 方法 使 用 === 来 检查 ， 看 
readdir() 的 返回 值 确实 不 为 false。 在 测试 中 ， 只 有 结果 0 才 会 得 到 false， 
因此 ， 我 们 避免 了 问题 。 


提示 : 


如 果 你 发现 目录 列表 中 的 项 目的 顺序 十 随意 的 ， 可 能 是 因为 由 文件 系统 上 自己 决定 顺序 。 
如 果 想 要 让 项 目 控 照 指定 的 方式 排列 ， 必 须 把 内 容 读 取 到 一 个 数组 中 ， 人 然后， 按照 想 要 的 方 
式 排序 它们 并 顺序 地 显示 。 


13.8 使 用 popen0 打 开 到 进程 和 离开 进程 的 管道 


在 本 章 前 面 ， 我 们 学 习 了 如 何 使 用 fopen0) 函 数 打开 一 个 文件 以 供 读 
取 和 写 入 。 现 在 ， 我 们 将 会 看 到 ， 可 以 使 用 popen0 〇 函数 来 打开 到 一 个 进 
TERI te 


popen) KIZ FJ LAG FEH - 


$file pointer = popen("some command", mode) 
WENA ger CER) wkw CBA) 。 


程序 清单 13.15 设计 用 来 产生 一 个 错误 ， 它 试图 打开 一 个 文件 以 供 
恋 取 ， 而 这 个 文件 并 不 存在 。 


程序 清单 13.15 ”使 用 popen0 来 读 取 一 个 文件 


: <?php 

: $file_handle = popen("/path/to/fakefile 2>&1", "r"); 
: $read = fread{$file handle, 2896); 

: echo $read; 

: pclose($file handle); 

?> 


OO AJO 一 


我 们 首先 在 第 2 行 调用 popen() 函 数 ， 试 图 打开 一 个 文件 以 供 读 取 。 
在 第 3 行 ， 读 取 存 储 在 $file_handle 指 针 中 的 任何 错误 消息 ， 并 在 第 4 行 显 
示 到 屏 妖 上 。 最 后 ， 第 5 行头 财 了 和 在 第 2 行 中 打开 的 文件 指针 。 如 末 把 这 
段 代 但 保存 为 test_popen.php， 将 其 放置 到 文档 根 目 录 下 ， 并 且 通 过 Web 
浏 磺 亏 访 问 它 ， 你 将 会 看 到 如 下 所 示 的 错误 消息 。 


The system cannot find the path specified. 


程序 清单 13.16 也 使 用 了 popeng0 来 谈 取 一 个 进程 的 输出 ， 在 这 个 例 
子 中 ， 是 UNIX 的 who 命 令 的 输出 。 


程序 清单 13.16 ”使 用 popen0) 读 取 UNIX who 命 令 的 输出 (只 适用 于 UNIX) 


<?php 
$handle = popen("who", "r"); 
while (!feof{$handle)) { 
$line = fgets($handle,1024); 
if (strlen{$line) >= 1) { 
echo $line."<br/>"; 
} 
} 
pclose($hand1e) ; 
Pe 
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© 


ERT, SA S RR EH popenQ hR, IRIE] S Ae 
指针 。 第 3 行 开始 了 一 个 while 循 坏 ， 它 会 从 进程 读 取 输出 的 每 一 行 ， 并 
日 最 终 在 代 公 的 第 6 行 显示 每 一 行 输出 。 如 琳 这 些 行 中 包含 有 信息 的 
话 ， 连 接 在 第 9 行 关闭 。 

如 果 把 这 段 代 码 保存 为 popen who.php， 并 将 其 放置 到 文档 根 目 录 
下 ， 当 使 用 Web 浏 览 夯 访问 它们 时 ， 你 将 会 看 到 如 下 所 示 的 内 容 《〈 当 
然 ， 这 和 你 的 实际 信息 而 不 是 我 的 实际 信息 相关 ) 。 
julie pts/® Mar 14 96:19 (adsl-63-206-12@-158.dsl.snfc21.pacbell.net) 

程序 清单 13.17 展 示 了 如 何以 与 入 模式 来 使 用 popen0， 从 而 把 数据 
传递 给 外 部 应 用 程序 。 这 个 例子 中 的 外 部 应 用 程序 叫做 column。 这 上段 脚 
本 的 目标 是 获取 多 维 数 组 的 一 个 元 系 ， 并 且 将 其 以 表 笠 的 形 陈 输出 到 一 
个 ASCII 文 件 中 。 


程序 清单 13.17 使 用 popen0O 把 数据 传递 到 UNIX column 命 令 〈 只 适用 于 UNIX ) 


1: <?php 

2: $products = array{ 

i i array{ "HAL 2000", 2, "red"), 

4: array("Tricorder", 3, "blue"), 

ac array{"GQRAC AI", 1, "pink"), 

6: array( "Sonic Screwdriver", 1, "orange" ) 
7: ); 

9: $handle = popen{"column -tc 3 -s / > jsomepath/purchases.txt", "w"); 
10: foreach ($products as $p) { 

113 fouts($handle, join{'/',$p)."\n"); 

12: 


: } 
13: pclose($handle} ; 
14: echo "done"; 
Let “eS 


在 第 2 行 到 第 7 行 ， 我 们 定义 了 一 个 名 为 $products 的 多 维 数组 ， 并 且 
在 其 中 放置 了 4 个 条 目 表 示 市 有 名 称 、 数 量 和 两 色 的 4 种 产品 。 在 第 9 
行 ， 使 用 popen0 以 写 入 格式 来 向 column 应 用 程序 发 送 一 条 命令 。 这 条 命 
令 癌 column 应 用 程序 及 送 参 数 ， 让 和 它 使 用 或 作为 字段 分 隔 符 ， 把 输入 格 
式 化 为 一 个 具有 3 列 的 表格 。 输 出 将 会 发 送 到 一 个 名 为 purchases.txt 的 文 
件 中 ， 请 确保 把 路 径 修 改 为 你 的 系统 上 存在 的 一 个 路 径 。 


在 第 10 行 到 第 12 行 ， 我 们 使 用 foreach Kii A Sproducts 数组 并 且 
把 每 个 元 素 及 送 给 打开 文件 指针 。 然 后 ， 用 join() 函 数 把 数组 转换 为 字 
Pe, TAMIR ENTE aha. RIER 13 行 天 闭 了 打开 文 
件 指 针 ， 并 且 在 第 14 行 把 一 条 状态 消息 显示 到 屏 攻 上 。 


如 果 我 们 把 这 段 代 码 保 存 为 popen_column.php， 将 其 放置 到 文档 根 
目录 下 ， 并 且 使 用 web 浏览 喜来 访问 它 ， 应 该 在 指定 的 位 置 中 创建 一 个 
文件 。 查 看 在 我 日 己 机 器 上 所 创建 的 文件 ， 可 以 看 到 如 下 文本 。 


HAL 2000 2 red 
Tricorder 3 blue 
ORAC AI 1 pink 
Sonic Screwdriver 1 orange 


你 的 系统 中 可 能 有 也 可 能 没有 column 程序 ， 但 是 本 节 说 明了 打开 
条 到 应 用 程序 的 管道 的 逻辑 和 语法 。 请 自行 采用 可 用 的 其 他 程序 来 党 
试 这 些 人 逻辑 。 


13.9 ”使 用 exec0 运 行 命 令 


exec() 函数 是 我 们 可 以 用 来 同 shell 传送 命令 的 几 个 函数 之 一 。 
exec() 闵 数 雷 要 一 个 字 从 串 并 且 可 选 地 接受 一 个 数组 变量 和 一 个 标量 变 
量 ， 字 符 串 表示 要 运行 的 命令 的 路 径 ， 数 组 变量 将 会 包含 命令 的 输出 ， 
标量 变量 将 包 售 返回 值 GEKO) ， 示 例如 下 。 


exec{"/path/to/somecommand", $output array, $return val}; 


程序 清单 13.18 使 用 exec() 函 数 和 基于 shell 的 ls 命令 来 产生 一 个 目录 
列表 。 


程序 清单 13.18 ”使 用 exec0O) 函 数 和 ls 生成 一 个 目录 列表 (只 适用 于 UNIX) 


: <?php 

: exec("1s -al .", $output array, $return val) ; 
echo "Returned ".$return val."<br/><pre>"; 

: foreach ($output_array as $o) { 

echo $o."\n"; 


echo "</pre>"; 
?> 


ON noah on 一 


在 第 2 行 ， 我 们 使 用 exec0O) 函 数 来 执行 Is 命令 。 我 们 把 命令 的 输出 
放 入 到 $output_array 数 组 ， 并 且 把 返回 值 放 入 到 变量 $return_val。 第 3 行 
只 十 显示 了 返回 全 ， 而 第 4 行 到 第 6 行 的 foreach 循 环 显 示 出 $output_array 
PN BETS IG 


fe 7s: 








第 3 行 中 的 字符 串 包含 了 一 个 开始 的 <pre> 标 记 ， 第 7 行 提供 了 一 个 结束 标记 。 这 只 是 通过 
使 用 HTML 预 处 理 的 文本 来 确 你 目录 列表 可 读 。 


如 果 把 上 述 代 码 保存 为 exec_ls.php， 将 其 放置 到 文档 根 目 录 中 ， 并 
日 使 用 Web 浏 览 占 访问 它 ， 将 会 看 到 如 图 13-8 所 示 的 结果 。 当 然 ， 这 和 


你 的 实际 信息 而 不 是 我 的 实际 


(SY tmp.html 


全 


Returned 0 

total 100 

GQIrwxr—-xXr-x 2 someusr 
drwxr-x-——— 7 someusr 
—-rw-r--r—— 1 someusr 
—-rw-r--r——- 1 someusr 
-rW-r--r—-- 1 someusr 
—-rw-r--r——- 1 someusr 
-rW-r--r—- 1 someusr 
-Irw-r--r-- 1 someusr 
-rWwW-r--r—- 1 someusr 
-rW-r--r—- 1 someusr 
-rW-r--r-- 1 someusr 
-rW-r--r-- 1 someusr 
-rW-r--r—-- 1 someusr 
-rW-r--r-- 1 someusr 
—-rw-r-—-r——- 1 someusr 
-rW-r--r-—- 1 someusr 
—-rw-r--r-——- 1 someusr 
—-rw-r--r-- 1 someusr 
-Irw-r--r-- 1 s0meusr 
一 工 何 一 工 一 一 工 一 一 1 sameusr 
一 工 休 一 工 一 一 工 一 一 1 sameusr 
一 工 何 一 工 一 一 工 一 一 1 someusr 
一 工 何 一 工 一 一 工 一 一 1 someusr 
一 工人 一 工 一 一 工 一 一 1 someusr 
一 工 休 一 工 一 一 工 一 一 1 someusr 


someusr 4096 
nobody 4096 
Someusr 162 
sSomeusr 763 
someusr 279 
someusr 18 
someusr 18 
someusr 18 
someusr 155 
someusr 23 
someusr Ja 
someusr 341 
someusr 165 
someusr 161 
someusr 220 
someusr 167 
someusr 166 
someusr 276 
someusr 123 
someusr 30 
someusr 33 
someusr 35 
someusr 131 
someusr G5 
someusr 341 

图 13-8 
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exec 153.php 
Filetests.php 
getbinary .php 
incfilel.txt 
inctile? .txt 
incfiles .txt 
loopy include.php 
myinclude .php 
myinclude? .php 
popen column.php 
popen who.php 
readchars .php 
readdir.php 
readlines .php 
readlines? .php 
readseek.php 
returnvalue.php 
TEST. CAG 

test include.php 
test include? .php 
test popen.php 
test returnvalue.php 
watile.php 


exec_ls.php 的 输出 


尽管 PHP 精 彩 纷 呈 ， 但 还 是 有 这 样 的 时 候 ， 即 我 们 需要 把 东 种 功能 
整合 到 基于 PHP 的 应 用 程序 中 ， 但 是 其 他 人 已 经 用 Pen 编号 了 做 同样 事 
情 的 代码 。 在 这 种 情况 下 ， 没 有 必要 再 重新 发 明 轮 子 ， 因 为 我 们 可 以 直 


接 使 用 exec0 来 访问 已 有 的 Perl 脚 本 并 利用 其 功能 。 然 和 而， 列 筷 了 了 ， 调 用 
一 个 外 部 应 用 程序 总 是 会 增加 脚本 的 负担 ， 无 论 从 时 间 上 还 是 从 内 存 使 
用 上 都 是 如 此 。 


13.10 ”使 用 system0 或 passthru0 运 行 命令 


system0 函 数 和 exec0O 国 数 类 似 ， 因 为 它们 都 局 动 一 个 外 部 应 用 程 
序 ， 并 且 它 使 用 一 个 标量 变量 来 存储 返回 值 ， 示 例如 下 。 


system("/path/to/somecommand", $return val}; 


system() EIA AlexecQRIAMN A EEF, EAR bias fa, 
而 没有 编程 的 和 干预。 如 下 的 代码 段 使 用 systemO0 回 man 命令 显示 一 个 
man 页 面 ， 这 个 页 面 使 用 <pre></pre> 标 记 对 格式 化 。 


<?php 

echo "<pre>" ; 

system{"man man | col -b", $return val}; 
echo "</pre>"; 

F> 


类 似 地 ，passthru0) 函 数 的 语法 和 system0) 函 数 相同 ， 但 行为 不 同 。 
使 用 passthru0 的 时 候 ，shell 命 令 的 任何 输出 在 返回 给 你 的 过 程 中 不 会 绥 
存 ， 这 适用 于 运行 产生 二 进 制 数 据 而 不 是 简单 的 文本 数据 的 命令 。 一 个 
例子 就 是 使 用 shell 工 具 来 定位 一 个 图 像 并 且 将 其 发 送 回 浏览 器 ， 如 程序 
清单 13.19 所 示 。 


程序 清单 13.19 ”使 用 passthru0 输 出 二 进 制 数据 


: <?php 
: if ((isset($ GET['imagename'])) && (file exists($ GET[ Imagename ]))) { 
header ( "Content-type: image/gif"); 
passthru("giftopnm ".$ GET['imagename']." | 
pnmscale -xscale .5 -yscale .5 | ppmtogif"); 


iD Oo OAR OND 一 


} else { 

echo "The image ".$ GET[ 'imagename']." could not be found"; 
} 
总 二 





这 个 脚本 中 使 用 的 shell 工 具 ， 包 括 giftopnm、pnmscale 和 ppmtogif， 也 可 能 在 你 的 系统 上 









需要 从 你 的 操作 系统 的 发 布 CD 上 安装 它们 ， 但 是 不 用 担 
通过 这 个 程序 清单 来 理解 使 用 passthru0 函 数 的 概念 。 





没有 安装 。 如 果 没 有 安装 ， 可 能 
Cy, 这 只 是 一 个 例子 。 这 里 5 只 是 





假设 把 这 个 文件 命名 为 getbinary.php， 它 将 会 像 下 面 这 样 从 HTML 
调用 。 


<img sre= getbinary.php?1imagename=<?php echo urlencode(*test.gif"} > > 


在 程序 清单 13.19 的 第 2 行 ， 测 斌 用户 输入 以 确保 涉及 到 的 文件 〈 根 
据 HIML 代 码 段 ， 是 文件 test.gift) 是 存在 的 。 由 于 这 个 脚本 将 会 癌 浏 抠 
侣 输出 GIF 数 据 ， 相 应 的 标 尖 在 第 3 行 设置 。 


在 第 4 行 和 第 5 行 ，passthru(0) 函 数 连续 执行 3 个 不 同 的 命令 ， 
giftopnm, pnmscale#ilppmtogif, ENTERADO JE Se rey FE M 8 a FE 
的 50%. passthru() Kihn th, tE ma AY A RAH A AIK BI I DBE o 


提示 : 





在 这 个 及 其 他 和 系统 相关 的 例子 中 ， 我 们 可 以 使 用 escapeshellcmd0O 或 escapeshellargO 函 数 
在 用 户 输入 中 转 义 元 素 。 这 cu 就 确保 了 用 户 不 会 诱 使 系统 执行 随意 的 命令 ， 例 如 删除 重要 
系统 文件 或 者 重新 设置 密码 。 这 些 函 数 提 供 了 用 户 输入 的 第 一 个 实例 ， 示 例如 下 。 


$new input = escapeshellcmd($ GET[ 'someinput']j}; 


然后 ， 我 们 可 以 在 脚本 中 的 其 他 地 方 引 用 $new_input 变 量 ， 而 不 是 $_GET [ 'someinput' ]. 
使 用 这 两 条 命令 ， 加 上 确保 编写 的 脚本 使 其 只 执行 我 们 和 希望 它 执行 的 任务 ， 而 不 是 执行 来 目 
用 户 的 命令 ， 以 上 残 是 确保 系统 安全 的 一 种 方法 。 


13.11 ”小结 


在 本 章 中 ， 我 们 学 习 了 如 何 使 用 include 等 语句 Cinclude_once(). 
reduire0 和 作为 额外 增添 的 require_once0 ) 来 把 文件 包含 到 文档 中 ， 并 
日 执行 包含 文件 中 的 任何 PHP 人 代码。 和 学习 了 如 何 使 用 攻 些 PHP 文 件 测 试 
函数 ， 以 及 用 来 按 行 、 按 字符 或 者 控 任 总 数据 量 读 取 文件 的 函数 。 和 学 习 
了 如 何 与 入 到 文件 ， 不 窒 是 伍 代 已 有 有 内 容 还 古 洪 加 a 到 已 有 内 容 的 后 面 ， 
并 且 我 们 学 习 了 如 何 创 建 、 移 动 和 读 取 目录 。 


我 们 还 学 习 了 和 系统 及 其 外 部 应 用 程序 通信 的 各 种 方法 。 尺 官 PHP 
是 快速 而 健壮 的 语言 ， 我 们 还 是 会 友 现 直接 利用 已 天 的 其 他 语言 (如 C 
或 Perl) 的 脚本 会 更 节约 成 本 和 时 间 。 我 们 可 以 使 用 popen()、exec()、 
system0 和 passthru0 函 数 访 问 这 些 外 部 应 用 程序 。 


我 们 学 习 了 如 何 使 用 popen() 把 数据 导出 给 一 条 命令 ， 对 于 从 标准 输 
入 接受 数据 的 应 用 程序 以 及 当 我 们 想 要 解析 由 应 用 程序 发 送 给 我 们 的 数 
据 的 时 候 ， 这 非常 有 有 用。 我们 还 学 习 了 使 用 execO0 和 systemO 把 命令 传递 
给 shell 以 及 获取 用 户 输入 。 我 们 还 学 习 了 使 用 passthru(0) 函 数 来 接受 作为 
一 条 shell 命 令 的 结果 的 二 进 制 数 据 。 


既然 可 以 更 多 地 使 用 文件 系统 ， 我 们 融 可 以 保存 和 访问 更 多 数量 的 
数据 。 然 而 ， 如 果 我 们 需要 从 大 的 文件 中 查找 数据 ， 这 样 的 脚本 会 变 得 
相当 慢 。 当 友 生 这 种 情况 的 时 候 ， 我 们 需要 使 用 一 个 数据 库 系 统 ， 稍 后 


将 介绍 它 。 


13.12 Q&A 


Q: include 语 句 会 使 脚本 变 慑 吗 ? 


A: 由 于 一 个 被 包含 的 文件 必须 由 引擎 打开 和 解析 ， 这 增加 了 负 
晶 。 然 而 ， 相 对 降低 的 性 能 负载 ， 可 章 用 代码 库 市 来 的 好 处 更 大 。 


a+ 


Q: 如 末 一 个 文件 无 法 打开 以 便 与 入 或 谈 取 ， 我 们 总 古 应 去 
结束 脚本 执行 吗 ? 


A: 应 该 总 是 考虑 到 这 种 可 能 性 。 如 采 脚 本 绝对 依靠 我 们 想 要 使 用 
的 文件 ， 你 可 能 布 望 使 用 die(0) 孙 数 ， 同 浏 贤 右 显 示 一 条 含有 信息 的 错误 
消 轧 。 在 重要 程度 较 低 的 情况 下 ， 我 们 仍然 需要 郑 夸 到 矢 效 ， 可 以 将 其 
写 入 到 一 个 日 志文 件 中 。 


Q: 在 网 上 哪里 可 以 获得 关于 安全 的 更 多 信息 ? 
A: 对 Web 安 全 的 一 个 权威 性 介绍 是 Lincoln Stein 所 编写 的 文档 


«The World Wide Web Security FAQ) ， 可 以 
7Ehttp://www.w3.org/Security/Faq/ 找到 . 


13.13 KRAJ 


这 个 实践 练习 设计 用 来 帮助 你 预测 可 能 的 问题 、 复 习 已 经 学 过 的 知 
识 ， 并 且 开 始 把 知识 用 于 实践 。 


问答 题 


1. 使 用 什么 函数 把 库 代 码 添加 到 当前 运行 的 脚本 中 ? 

2. 使 用 什么 函数 来 确认 一 个 文件 是 否 在 你 的 文件 系统 上 ? 
3. 如 何 确 定 一 个 文件 的 大 小 ? 

4. 使 用 什么 函数 来 打开 一 个 文件 以 供 读 取 和 写 入 ? 

5. 使 用 什么 函数 来 从 一 个 文件 读 取 一 行 数据 ? 

6. 如 何 获知 到 达 一 个 文件 的 末尾 ? 

7. 使 用 什么 函数 来 把 一 行 数据 写 入 到 一 个 文件 ? 

8. 如 何 打开 一 个 目录 以 供 读 取 ? 


9， 在 我 们 打开 一 个 目录 之 后 ， 使 用 什么 函数 来 读 取 一 个 目录 项 的 
BF 


10. EHAA eR BOR FT FF a TE YJ? 


11. 当 我 们 打开 一 个 连接 之 后 ， 如 何 从 一 个 进程 读 取 数 据 ? 写 入 数 
据 呢 ? 


12. 在 把 用 户 输 入 传递 到 一 个 shell 命 令 之 前 ， 如 何 转 义 用 户 输 入 使 
其 更 安全 一 些 ? 


oy 


EE 
1. 我 们 可 以 使 用 require0) 或 includeO 语 名 来 把 PHP 文 件 加 入 到 当前 
的 文档 中 ， 也 可 以 使 用 include_once0) 或 require_once()。 
2. 我 们 可 以 使 用 名 e_existsO 函 数 来 测试 一 个 文件 的 存在 性 。 
3. flesize0 函 数 返 回 一 个 文件 的 字 节 大 小 。 


4. fopen0 函 数 打 开 一 个 文件 。 它 接受 一 个 文件 的 路 径 以 及 一 个 表 
示 模 式 的 字符 。 它 返回 一 个 文件 源 。 


5. fgets() 孙 数 读 取 数据 耳 到 达到 我 们 传递 给 它 的 缓冲 大 小 、 或 者 到 
达 行 的 末尾 ， 或 者 到 达 文 档 的 末尾 ， 不 党 上 述 哪 种 情况 ， 以 先 出 现 的 为 
准 。 


6. 当 传 隶 给 feof() 函 数 的 文件 源 a 到 达 文 件 末尾 的 时 候 ，feof0) 函 数 返 


[A] true . 
7. RATE DAE H fputs() R BOR IE Bs BS APS IF o 
8. opendirO 函 数 允 许 我 们 打开 一 个 目录 以 供 读 取 。 
9. readdir0 函 数 返 回 一 个 打开 的 目录 中 的 一 个 目录 项 的 名 字 。 
10. popenO 函 数 用 来 打开 到 一 个 进程 的 管道 


11. 我 们 可 以 从 一 个 进程 谈 取 数据 或 把 数据 与 入 一 个 进程 ， 融 好 像 


我 们 使 用 一 个 打开 的 文件 一 样 ， 即 使 用 feofO0 和 fgets0 来 谈 取 ， 使 用 
fputs(0) 来 写 入 。 


12. 如 果 用 户 输 入 是 你 的 Shell 命令 的 一 部 分 ， 可 以 使 用 
escapeshellcmd() 或 escapeshellarg() KAŽ KE HEFE XEN]. 


FAG el 


: 创建 一 个 表单 ， 它 接 有 党 一 个 用 户 的 第 一 个 名 宇和 第 二 个 名 字 。 
te 文 些 数 据 剑 存 到 一 个 文件 。 


2. 创建 一 个 脚本 ， 读 取 我 们 在 思考 题 1 中 创建 的 数据 文件 。 除 了 
将 其 内 容 显示 到 浏览 器 〈 在 每 行 添加 一 个 <bv> 标 记 ) ， 还 显示 出 包 售 
文件 中 的 行 数 和 文件 大 小 的 摘要 。 


第 14 革 ”使 用 图 像 


在 本 章 ， 我 们 将 学 到 如 下 扩容 : 


。 如 何 修改 PHP 来 增加 和 图 形 相关 的 功能 。 
。 如 何 创建 一 个 新 的 图 像 。 
。 如 何 修改 已 有 的 图 像 。 


PHP 的 标准 安 闻 拥 有 很 多 内 建 的 函数 ， 可 以 用 来 动态 地 创建 和 操作 
图 像 。 币 见 用 法 包括 图 示 的 创建 ， 以 及 修改 已 有 的 图 像 来 好 示 水 印 。 只 
南 要 略 作 调整 ， 我 们 残 可 以 更 多 地 扩展 这 些 图 数 。 在 本 章 中 ， 我 们 学 习 
使 用 PHP 函 数 创建 和 操作 图 像 的 基础 。 


14.1 理解 图 像 创建 过 程 


使 用 PHP 创 建 一 个 岁 像 ， 不 像 使 用 一 个 绘图 程序 创建 图 人 乔 那样 〈 例 
如 ，Sumo Paint. Corel DRAW 或 Windows Draw) 。 这 里 没有 指 问 、 点 
击 或 拖 搜 闫 料 桶 到 一 个 预先 定义 的 空间 以 填 元 图 像 竺 操作。 同样 ， 这 里 
也 没有 另存 为 功能 ， 使 绘图 程序 目 动 创建 一 个 GIF、JPEG、PNG 等 ， 仅 
仅 因 为 我 们 要 求 它 这 么 做 。 


相反 ， 我 们 必须 变 成 一 个 绘图 应 用 程序 。 作 为 程序 员 ， 我 们 必须 告 
诉 PHP5| 敬 在 绘图 过 程 中 的 每 一 步 做 什么 。 我 们 负 贡 使 用 单个 的 PHP 孙 
数 来 定义 闫 色 、 绘 制 和 填 元 形状 、 确 定 和 调整 图 像 大 小 ， 并 且 把 图 像 你 
存 为 一 个 指定 的 文件 类 型 。 然 而 ， 如 果 你 理解 这 个 过 程 中 的 每 一 步 ， 并 
且 按 照 顺 序 完 成 任务 的 话 ， 这 并 不 像 看 上 去 那么 难 。 


VERS: 








AS ePTFE AS xe PHP BEARERS ASIII, (Ee, Be EDR TB], ASS IE CHAR 
能 够 介绍 这 些 内 容 。 我 们 可 以 保证 ， 对 于 使 用 PHP 中 与 图 像 相关 的 函数 ， 这 些 例子 将 会 给 你 打 
下 很 好 的 基础 。 


大 村 颜色 


当 在 图 像 相 关 的 脚本 中 定义 颜色 的 时 候 ， 我 们 将 使 用 RGB 颜色 系 
统 。 针 对 红色 、 绿 色 和 蓝 色 (R、G 和 了 B) 中 的 每 一 项 ， 使 用 从 0 到 
255 之 间 的 十 进 制 数 值 ， 这 样 就 可 以 定义 一 个 具体 的 颜色 了 。0 表 示 没 
有 任何 数量 的 该 颜色 ， 而 255 表 示 该 颜色 的 最 大 量 。 


例如 ， 纯 红色 的 RGB 值 是 (255，0，0) ， 或 者 说 完全 分 配 红 色 
值 ， 而 没有 绿色 和 葛 色 。 类 似 的 ， 纯 绿色 的 值 为 (0，255，0) ， 纯 蓝 
AEA (0, 0, 255). AAT, AfIRGBIBW (255, 255, 
255) ， 而 黑色 的 RGB 值 为 (0，0，0) 。 漂 亮 的 暗 紫 色 的 RGB 值 是 
(153, 51, 153) ， 而 浅 灰 色 的 RGB 值 是 (204，204，204) 。 为 了 方 
便 参 考 ， 可 以 查看 http://en.wikipedia. org/wiki/Web_colors 的 两 色 和 RGB 
值 的 列表 。 


14.2 ”对 PHP 的 必要 修改 


PHP 的 当前 发 布 版 本 包含 了 Thomas Boutell 的 GD 图 形 库 的 一 个 绑 定 
版 本 。 包 含 了 这 个 库 也 束 不 再 需要 下 载 和 安装 第 三 方 的 库 了 ， 但 是 这 个 
fee Tig Be TE ZZ A EN EN (RI o 


要 在 安装 的 时 候 使 得 GD 库 可 用 ，Linux/UNIX 用 户 必 须 在 准备 编译 
PHP 的 时 候 同 配置 参数 添加 如 下 内 容 。 
- -with-gd 

再 次 运行 PHP 配 置 程序 之 后 ， 你 必须 像 第 4 章 所 介绍 的 那样 使 用 一 
次 make 和 make install tE. BEH GDH Windows H P W Ta 
php_gd2.dll 作 为 php.ini 文 件 中 的 一 个 扩展 ， 正 如 我 们 在 第 4 章 所 介绍 的 。 


当 使 用 GD 库 的 时 候 ， 除 非 我 们 安 友 了 其 他 的 库 ， 人 否则 被 限制 只 能 
使 用 GIF 格式 的 文件 。 


使 用 GIF 文件 能 够 很 好 地 满足 我 们 的 需要 ， 但 是 ， 如 果 我 们 想 要 在 
Linux/UNIX 上 创建 JPEG 或 PNG 文件 ， 就 需要 下 载 并 安装 一 些 其 他 
库 ， 并 且 对 我 们 的 PHP 安装 做 一 些 修改 “Windows 平 台 上 的 安装 中 包含 
了 这 些 库 并 且 可 用 ) 。 


。JPEG 库 及 其 信息 可 以 在 http://www.ijg.org/ 找到 。 

。PNG 库 及 其 信息 可 以 在 http://www.libpng.org/pub/png/libpng.html FX 
到 。 

。 如 东 想 要 使 用 PNG 文 件 ， 还 应 该 安 儿 zlib 库 ， 可 以 


7Ehttp://www.zlib.net/ 找到 。 


按照 这 些 站 点 上 的 说 明 来 安装 这 些 库 。 在 安装 之 后 ，Linux/UNIX 用 
户 必 须 通过 把 如 下 的 内 容 添 加 到 PHP 配 置 参数 中 ， 从 而 再 次 配置 和 重新 
编译 PHP (假设 我 们 想 要 使 用 所 有 3 个 库 ， 如 果 不 是 的 话 ， 只 需要 添加 
需要 使 用 的 库 ) 。 


--with-jpeg-dir=[path to jpeg directory] 
--with-png-dir=[path to PNG directory] 
--with-zlib=[path to zlib directory] 


再 次 运行 PHP 配 置 程序 之 后 ， 你 必须 像 第 4 章 所 介绍 的 那样 使 用 一 
VX make 和 make install 的 过 程 。 这 样 ， 我 们 的 库 束 油 活 了 并 准备 好 使 用 
J: 


14.3 ”绘制 一 个 新 的 图 像 


用 来 创建 一 个 新 图 像 的 基本 的 PHP 函 数 叫 做 ImageCreate0， 但 是 ， 
创建 一 个 图 像 并 不 像 调用 函数 那么 简单 。 创 建 一 个 图 像 是 一 个 逐步 的 过 
往 ， 并 且 包 括 几 种 不 同 的 PHP 函 数 的 使 用 。 


AE —“ A&M ImageCreate()ei BF aa, (Ase, SER AAT VERY A 
Fe At RS A HH Be DX eR RE TRS ES “S300 Ke FE. 300 
(RAR m A E XE 


$myImage = ImageCreate(300,300) ; 


INTE RE MU ER, Be ROR AT AE MH i Bo A BE o 
如 下 的 例子 使 用 ImageColorAllocateO) 函 数 和 RGB 值 定 义 了 5 种 这 样 的 凑 
E (Ale Re. A. AE, RRA) 。 


$olack = ImageColorAllocate($myImage, @, @, @); 
$Swhite = ImageColorAllocate($myImage, 255, 255, 255}; 
$red = ImageColorAllocate({$myImage, 255, $, @); 
Sgreen = ImageColorAllocate($myImage, @, 255, 0}; 
$blue = ImageColorAllocate($myImage, 0, @, 255); 


提示 : 








在 你 的 脚本 中 ， 所 分 配 的 第 一 种 颜色 用 作 图 像 的 背景 闫 色 。 在 这 个 例子 中 ， 背 景 闫 色 是 
黑色 (你 也 可 以 根据 使 用 环境 ， 将 $background_coloror 变 量 命名 为 其 他 有 意义 的 名 字 ) 。 


既然 我 们 了 解 了 设置 绘图 画布 和 分 配 闫 色 的 基础 知识 ， 现 在 可 以 继 
续 学 习 绘 制 形状 并 且 真 正 地 把 图 像 输 出 到 一 个 Web 浏 贤 伏 了 了。 


14.3.1 绘制 形状 和 线条 


AU BJ LAS PHP ek 2 A BER Ee HH LE ZH ZR FZ 


e ImageEllipseO 用 来 绘制 一 个 顶 圆 。 

。 ImageArc() 用 来 绘制 一 个 部 分 椭圆 。 

e ImagePolygon0 用 来 绘制 一 个 多 边 形 。 
。 ImageRectangle() 用 来 绘制 一 个 矩形 。 
。 ImageLine() 用 来 绘制 一 个 线条 。 


使 用 这 些 函 数 寅 要 先 所 前 考虑 一 下 ， 因 为 我 们 必须 设置 所 进行 的 给 
制 从 哪 一 点 开始 到 哪 一 点 结束 。 这 些 函 数 中 的 每 一 个 都 使 用 x 轴 绘 标 和 y 
轴 华 标 来 表示 在 画布 上 绘制 的 起 点 。 我 们 还 必须 定义 要 绘制 到 离 x 轴 和 y 
轴 多 远 的 位 置 。 

例如 ， 如 下 代码 在 男 布 上 绘制 了 一 个 定形 ， 充 定形 从 点 (15,15) 开 
人 ， 并 且 水 平方 同 80 个 像 系 ， 垂 直方 网 140 个 像 系 的 长 度 ， 因 此 ， 线 条 
在 点 (95,155) 结 束 。 万 外 ， 这 些 线条 将 使 用 红色 绘制 ， 访 颜色 已 经 在 变 
量 $red 中 定义 了 。 

ImageRectangle($myImage, 15, 15, 95, 155, $red}; 

BAR AE Be Zee ll KD A TF (ER AR EN A ERS TE AE 
MEIGS Pa, AY EAS STB JARI < 
ImageRectangle($myImage, 95, 155, 175, 295, $white); 

最 后 ， 为 了 请 加 与 第 一 个 窍 形 对 齐 的 其 他 红色 矩形 ， 但 是 ， 从 日 色 
定形 的 最 右边 的 所 开始 ， 使 用 如 下 代码 。 


ImageRectangle(SmyImage, 175, 15, 255, 155, $red}; 


Ferrin 14.129 S Aa AILIN—-TA RIERA, Js SL 


ITRI H R FE RNS fa LE BE Web ii wS o 
程序 清单 14.1 创建 一 个 新 的 图 像 


<?php 


{create the canvas 
$myImage = ImageCreate(30@, 300) ; 


//set up some colors for use on the canvas 

$black = ImageColorAllocate($myImage, @, @, 0); 
$white = ImageColorAllocate($myImage, 255, 255, 255); 
$red = ImageColorAllocate{$myImage, 255, 0, 0); 

9: $green = ImageColorAllocate($myImage, 0, 255, 0); 

18: $blue = ImageColorAllocate{$myImage, ð, @, 255); 


ONO fh ON 一 


12: //draw some rectangles 

13: ImageRectangle($myImage, 15, 15, 95, 155, $red); 

14: ImageRectangle(SmyImage, 95, 155, 175, 295, $white}); 
15: ImageRectangle($myImage, 175, 15, 255, 155, $red); 
17: //output the image to the browser 

18: header ("Content-type: image/png"); 

19: ImagePng($myImage) ; 

21: //clean up after yourself 


22: ImageDestroy($myImage) ; 
2a ‘te 


党 18 行 到 第 19 行 把 图 像 数 据 的 法 输出 到 Web Wiis, “EA eA 
所 创建 图 像 的 MIME 类 型 来 有 友 壕 相应 的 header0) 函 数 。 然 后 ， 使 用 
ImageGif()、ImageJpeg() 或 ImagePng0 函 数 相 应 地 输出 数据 流 ， 这 个 例 
子 输 出 的 是 一 个 PNG 文 件 。 在 第 22 行 ， 我 们 使 用 ImageDestroy() 函 数 清 
空 ImageCreate0O 国 数 在 脚本 开头 所 使 用 的 内 存 。 


把 上 述 程序 清单 保存 为 magecreate.php 并 且 将 其 放置 在 web 服务 亏 的 
文档 根 目录 下 。 访 问 它 的 时 候 ， 结 果 看 上 去 如 图 14-1 所 示 ， 只 不 过 还 带 
A ME o 


EY imagecreate.php (300x300) 


< C © hitp://localhost/14/imagecreatephp | 图 A & 








图 14-1 带 有 三 个 绘制 的 矩形 的 画布 
14.3.2 ”使 用 闫 色 填 充 


程序 清单 14.1 的 输出 只 是 生成 了 矩形 的 边框 。PHP 还 有 如 下 专门 
的 图 像 函数 ， 是 设计 用 来 填充 区 域 的 。 


e ImageFilledEllipseO) 用 来 填充 一 个 椭圆 。 

e ImageFilledArc() KKK Fe Th op A 

e ImageFilledPolygonO 用 来 填充 一 个 多 边 形 。 
e ImageFilledRectangle() 用 来 填充 一 个 矩形 。 


这 些 函 数 的 用 法 和 它们 对 应 的 非 填 充 函 数 的 用 法 是 相同 的 。 在 程序 
清单 14.2 中 ， 把 非 填 充 函 数 蔡 换 为 设计 用 来 填充 一 个 区 域 的 函数 。 换 名 
话说 ， 只 有 第 13 行 和 第 15 行 代码 修改 了 。 


© N oar Wh 一 


WO 


程序 清单 14.2 ”使 用 颜色 填充 创 建 一 个 新 的 图 像 


<?php 
//create the canvas 
$myImage = ImageCreate(300, 300) ; 


/iiset up some colors for use on the canvas 

$black = ImageColorAllocate($myImage, @, ©, @); 
$white = ImageColorAllocate($myImage, 255, 255, 255); 
$red = ImageColorAllocate($myImage, 255, @, @); 
$green = ImageColorAllocate($S$myImage, @, 255, Q); 


: $blue = ImageColorAllocate({$myImage, ð, @, 255); 


//draw some rectangles 

ImageFilledRectangle({$myImage, 15, 15, 95, 155, $red); 
ImageFilledRectangle({$myImage, 95, 155, 175, 295, $white); 
ImageFilledRectangle{$myImage, 175, 15, 255, 155, $red); 


//output the image to the browser 


: header ( "Content-type: image/png"); 


ImagePng ($myImage) ; 


//clean up after yourself 
ImageDestroy({$myImage) ; 


; i 


把 上 述 程序 清单 保存 为 imagecreatefill.php 并 且 将 其 放置 到 Web 服 务 


侣 的 文档 根 目录 下 。 访 问 它 的 时 候 ， 将 会 得 到 如 图 14-2 所 示 的 结 米 ， 了 也 
是 市 有 闫 色 的 。 


as) imaqgecreatefill.php (300% 5 


i C | Q hittp://localhost/14/imagecreatefill. 





图 14-2 WA ZARNA AY EZ EN A 


1444 ”绘制 有 趣 的 饼 图 


前 面 的 例子 有 挟 档 各 ,但 是 它们 回 你 介绍 了 创建 图 像 的 过 程 ， 包 括 
定义 田 布 、 定 义 闫 色 ， 然 后 绘制 和 填充 。 我 们 可 以 按照 同样 的 事件 顺序 
来 扩展 脚本 ， 针 对 数据 点 使 用 静态 或 动态 的 数据 从 而 创建 图 表 。 程 序 清 
单 14.3 绘制 了 一 个 基本 的 饼 图 。 第 1 行 到 第 10 行 看 上 去 和 前 面 的 程序 
清单 完全 相同 ， 因 为 它们 只 是 设置 画布 的 大 小 和 所 用 的 两 色 。 


程序 清单 14.3 ”一 个 基本 的 饼 图 


<?php 
//create the canvas 
$myImage = ImageCreate(300, 300) ; 


//set up some colors for use on the canvas 

$white = ImageColorAllocate($myImage, 255, 255, 255); 
$red = ImageColorAllocate{$myImage, 255, @, Q@); 
$green = ImageColorAllocate($myImage, @, 255, 0); 

9: $blue = ImageColorAllocate{$myImage, ð, ©, 255); 


ON nah WN 一 


11: //draw a pie 
12: ImageFilledArc($myImage, 100, 100, 200, 150, ©, 90, $red, IMG ARC PIE); 
13: ImageFilledArc(SmyImage, 100, 100, 200, 150, 90, 180 , $green, 
IMG_ARC PIE); 
14: ImageFilledArc($myImage, 100, 100, 200, 150, 180, 360 , $blue, 
IMG ARC PIE); 


16: //output the image to the browser 
17: header ("Content-type: image/png"); 
18: ImagePng(SmyImage) ; 

20: //clean up after yourself 


21: ImageDestroy($myImage) ; 
eon ee 


Hea iE MAE A, A A ee A EN. 


在 第 12 行 到 第 14 行 ， 我 们 使 用 了 ImageFilledArc(0) 函 数 ， 它 有 如 下 几 


个 属性 。 


。 图 像 的 标识 符 。 

。 部 分 李 圆 的 中 心 扣 x 坐 标 。 
。 部 分 李 圆 的 中 心 反 y 坐 标 。 
So APE Lea] EY TE BE o 

So APE Lea] AS tt JB o 

Bo APH LEAL ES FP Ue R o 
BANAR ES ER A o 

。 HE., 

。 位 式 。 


© 
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程序 清单 14.3 的 第 14 行 绘制 了 一 个 弧 形 。 


这 个 弧 形 应 该 使 用 定义 的 颜色 $blue 填充 ， 并 且 应 该 使 用 
IMG_ARC PIE 样式 。IMG ARC _PIE 样 式 是 用 来 显示 的 几 种 内 建 样式 
之 一 ， 这 个 样式 表示 创建 一 个 圆 角 的 边 。 


提示 : 





我 们 可 以 在 http://www.php.net/image 的 PHP 手 册 中 了 解 所 有 的 不 同样 式 。 


把 上 述 程序 清单 保存 为 imagecreatepie.php， 并 且 将 其 放置 到 Web 
服务 器 的 文档 根 目录 下 。 当 访问 它 的 时 候 ， 应 该 看 到 如 图 14-3 所 示 的 结 
R, Be A ae 





a) Imagecreatepie.php (300 - 
< CG |© http://localhost/14/imagecreatepie.php Ki nA” 





图 14-3 PAPER “Pe 


可 以 扩展 程序 清单 14.3 中 的 代码 并 得 到 一 个 3D 外 观 的 饼 图 。 为 了 实 
现 这 一 点 ， 为 边 定义 另外 3 种 颜色 。 这 些 颜 色 可 以 比 基 本 颜色 深 或 浅 ， 
只 要 它们 能 够 提供 某 种 对 比 。 下 面 的 例子 定义 了 3 种 较 浅 的 颜色 。 


$lt_red = ImageColorAllocate($myImage, 255, 150, 150}; 
$lt_green = ImageColorAllocate($myImage, 150, 255, 150); 
$lt_blue = ImageGolorAllocate($myImage, 150, 150, 255}; 


要 创建 阴影 效果 ， 我 们 使 用 一 个 for 循 环 从 点 (100, 110) 到 
(100, 101) 和 深 加 一 系列 较 小 的 跌 形 ， 使 用 较 浅 的 凑 色 作为 填充 两 色 。 


for ($i = 110;$i > 190;$i--} { 
ImageFilledArc (SmyImage, 180, $i, 200, 150, ð, 98, $lt red, IMG_ARC PIE}; 
ImageFilledArc ($myImage, 100, $i, 200, 150, 90, 180, $1t green, 
IMG ARC PIE}; 
ImageFilledArc ($myImage, 100, $i, 200, 150, 180, 360, $1t_blue, 
IMG ARC PIE}; 


程序 清单 14.4 给 出 了 绘制 一 个 3D 饼 图 的 代码 。 
程序 清单 14.4 ”一 个 3D 饼 图 


<?php 
{/create the canvas 
$myImage = ImageCreate (300, 380) ; 


/{/set up some colors for use on the canvas 

$white = ImageColorAllocate($myImage, 255, 255, 255); 
7: $red = ImageColorAllocate{$myImage, 255, @, 0); 

8: $green = ImageColorAllocate(SmyImage, @, 255, 0); 

9: $blue = ImageColorAllocate($myImage, @, ©, 255); 

10: $lt_red = ImageColorAllocate($myImage, 255, 150, 150); 
11: $lt green = ImageColorAllocate($myImage, 150, 255, 150); 
12: $lt blue = ImageColorAllocate($myImage, 150, 150, 255); 
13: 

14: //draw the shaded area 

15: for ($i = 110;$i > 10@;$i--) { 


Oooh &D NH — 


1G: ImageFilledArc ($myImage,100,$1,200,150,0,90,$1t_ red,IMG ARC PIE); 

T ImageFilledArc ($myImage,100,$1,200,150,90,180,$l1t_green,IMG ARC PIE); 
18: ImageFilledArc ($myImage, 100,$1i1,200,150,180,360,$1lt_ blue,IMG_ARC PIE); 
19: } 

20: 


21: //draw a pie 

22: ImageFilledArc($myImage, 100, 100, 200, 150, ©, 90, $red, IMG ARC PIE); 

23: ImageFilledArc($myImage, 100, 100, 200, 150, 90, 180 , $green, IMG_ARC_PIE)}; 
24: ImageFilledArc($myImage, 100, 100, 200, 150, 180, 360 , $blue, IMG_ARC PIE}; 
eas 

26: //output the image to the browser 

27: header ("Content-type: image/png"); 

28: ImagePng(S$myImage) ; 

29: 

30: //clean up after yourself 

31: ImageDestroy($myImage) ; 

32: T> 


把 上 述 程序 清单 保存 为 imagecreate3dpie.php， 并 且 将 其 放置 到 Web 
服务 器 的 文档 根 目 录 下 。 当 访问 它 的 时 候 ， 应 该 看 到 如 图 14-4 所 示 的 结 
R, DEWAR. 


a) Imagecreatesdpie.php (300 


€ C © http:;//localhost/14/imagecreate3dpie.php | 图 K 7 





图 14-4 ”一 个 带 有 分 块 的 3D 饼 图 


这 只 十 展示 攻坚 图 像 绘制 和 项 充 函数 的 功能 的 基本 例子 。 在 下 一 区 
中 ， 我 们 学 习 如 何 操作 已 有 图 像 。 


145 ”修改 己 有 图 像 


从 其 他 图 像 创建 图 像 的 过 程 ， 和 创建 一 个 狐 图 像 避 从 同样 的 基本 步 
台 ， 不 同 之 处 在 于 谁 充当 图 像 画 布 。 在 表面， 我 们 使 用 ImageCreate() 也 | 
数 来 创建 一 个 新 的 画布 。 当 从 一 个 已 有 图 像 创建 图 像 的 时 候 ， 我 们 使 用 
ImageCreateFrom*() #4!) PI BX © 


我 们 可 以 从 已 有 的 GIF、JPEG、PNG 以 及 各 种 各 样 其 他 的 图 像 类 型 
来 创建 图 像 。 从 这 些 格 式 来 创建 图 像 的 函数 是 ImageCreateFromGif()、 
ImageCreateFromJpg() 和 ImageCreateFromPng() 等 等 。 在 下 一 个 例子 中 ， 
你 可 以 看 到 从 一 个 已 有 的 图 像 创建 一 个 新 的 图 像 古 多 么 的 容易 。 图 14-5 
给 出 了 基本 图 保 。 


程序 清单 14.5 展 示 了 如 何 使 用 已 有 的 图 像 作为 画布 ， 然 后 在 其 上 给 
制 一 个 椭圆 。 
[=| james 
© baseimage.png (350x200) 


€ C | © http://localhost/14/baseimage.png I KoA 


This is my base image. 





图 14-5 ”基本 图 像 
程序 清单 14.5 “从 已 有 图 像 创 建新 图 像 


1: <?php 

2: /fuse existing image as a canvas 

3: $myImage = ImageCreateFromPng{ "baseimage.png"); 

4: 

5: f/fallocate the color white 

6: $white = ImageColorAllocate($myImage, 255, 255, 255); 
Ts 

8: rdraw on the new canvas 

9: ImageFilledEllipse($myImage, 180, 70, 20, 20, Swhite) 


; 
19: ImageFilledEllipse($myImage, 175, 78, 20, 20, $white); 
11: ImageFilledEllipse($myImage, 250, 70, 20, 20, $white); 


13: /foutput the image to the browser 
14: header ( "Content-type: image/png"}; 
15: ImagePng($myImage) ; 

17: //clean up after yourself 


18: ImageDestroy($myImage) ; 
Vea ee 


把 上 述 程序 清单 保存 为 imagefrombase.php， 并 且 将 其 放置 到 Web 服 
侨 鸭 文档 根 目录 下 。 当 访问 它 的 时 候 ， 将 会 看 到 如 图 14-6 所 示 的 结 


m R 


torn el] 
a) imagetrombase.php (350%. 


= CC © hitp://localhost/14/imagefrombase.php | 加 A 9 


This is my base image. 








图 14-6 ”在 已 有 图 像 上 绘制 图 像 


下 一 个 例子 把 这 个 过 程 回 前 推进 了 几 个 步骤 ， 并 且 使 用 了 一 些 不 同 
的 图 像 修改 函数 。 在 这 个 例子 中 ， 已 有 的 图 像 是 4 个 PNG 图 像 ， 每 一 个 
都 是 在 一 个 灰色 背景 上 有 一 个 不 同 颜 色 的 三 角形 分 区 。 在 程序 清单 14.6 
中 ， 在 每 一 个 步 又 中 ， 我 们 将 会 把 这 些 图 像 中 的 每 一 个 分 别 铬 加 在 男 一 
个 之 上 并 将 它们 组 合 起 来 ， 以 便 灰 色 背 景 变 成 透明 的 ， 并 且 其 下 方 的 网 
像 完 全 透 过 它 显 示 出 来 。 

在 第 3 行 ， 选 择 了 一 个 图 像 作 为 基本 图 像 。 在 这 个 例子 中 ， 它 是 
imgl.png。 第 8 行 到 第 1177 HY for 循环 中 处 理 了 大 量 的 工作 。 已 经 知道 
了 你 有 4 个 图 像 并 且 第 一 个 图 像 已 经 用 作 基 本 图 像 ， 那 么 ， 剩 下 的 其 他 
3 个 图 像 用 来 辣 加 并 且 变 得 透明 。 


程序 清单 14.6 ”从 加 图 像 并 且 使 它们 变 得 透明 


<?php 
//select an image to start with 
$baseimage = ImageCreateFromPng("img1.png"); 


/{loop through images #2 through the end 
for($i=2; $i <5; $i++) { 
/fallocate the transparent color, and stack 
$myImage = ImageCreateFromPng("img".$i.".png"); 
$gray = ImageColorAllocate{$myImage, 185, 185, 185); 
10: ImageColorTransparent($myImage, $gray) ; 
1 ImageCopyMerge($baseimage, $myImage,8@,0,0,0,150,150,120); 
12: } 


O N oath Wh — 


<a) 


14: //output the image to the browser 
15: header ( "Content-type: image/png"); 
16: ImagePng($baseimage} ; 


8: //clean up after yourself 
19: ImageDestroy({$baseimage) ; 
20: ?> 


在 第 8 行 创建 一 个 新 层 之 后 ， 其 灰色 区 域 表示 为 透明 的 ， 并 且 它 被 
合并 到 基本 图 像 的 上 面 。 当 这 个 层 登 加 后 ， 基 本 图 像 包 侣 一 个 递增 的 图 
层 编 亏 ， 百 到 达到 4 层 的 总 数 。 图 像 在 第 15 行 到 第 16 行 被 有 友 送 到 浏 顺 


BH 


AN o 


把 上 述 程序 清单 保存 为 imagestacker.php 并 且 将 其 放置 到 Web 服务 
侣 有 的 文档 根 目 录 下 。 当 访问 它 的 时 候 ， 将 会 看 到 如 图 14-7 所 示 的 结果 。 


A (SY imagestacker.php (150x150 x TY 


SS © hitte:/flocalhost/14/imagestacker php Ws 





图 14-7 eA A BR re 2 AR 


14.6 ”使 用 来 自用 户 输 入 的 图 像 创 建 图 像 


除了 从 其 他 图 像 创建 图 像 以 及 绘制 目 己 的 图 像 之 外 ， 我 们 还 可 以 根 
据 用 户 输入 来 创建 图 像 。 这 和 如 何 创建 脚本 没有 根本 区 别 ， 只 不 过 我 们 
需要 从 一 个 表 蛙 收集 值 ， 而 不 是 将 值 直接 编码 到 脚本 中 。 

在 程序 清早 14.7 中 ， 我 们 创建 了 一 个 一 体 化 的 表单 和 脚本 ， 它 要 求 
用 户 输 入 从 图 像 大 小 到 文本 其 色 以 及 背景 其 色 的 各 种 属性 ， 还 要 求 输入 
一 个 消 奶 字符 串 。 我 们 将 会 学 习 imagestring() 孙 数 ， 它 用 来 把 一 个 字符 
串 “ 写 入 ”到 一 个 图 像 上 。 

让 我 们 来 看 看 脚本 ， 第 2 行 到 第 38 行 表示 用 户 输入 表单 ， 其 他 的 
代码 行 根 据 用 户 的 要 求 处 理 所 创 建 的 图 像 。 


程序 清单 14.7 MAP A ET AR 


OOnN Oa hLGAN = 


<?php 
if (!% POST) { 
//show form 
?> 
<!DOCTYPE html> 
<html> 
<head> 
<title>Image Creation Form</title> 


<style type="text/css"> 

fieldset{border: 0; padding: Opx Opx 12px @px;} 
fieldset label {margin-left: 24px;} 

legend, label {font -weight:bold;} 

</style> 


</head> 
<body> 
<hi>Create an Image</h1> 


<form method="POST" action="<?php echo $ SERVER['PHP_SELF']; ?>"> 


<fieldset> 

<legend>Image Size:</legend><br/> 
<label for="w">W:</label> 
<input type="text" id="w" 
<label for="h">H:</label> 
<input type="text" id="h" name="h" size="5" maxlength="5" /> 
</fieldset> 


name="w" size="5" maxlength="5" /> 


<fieldset> 

<legend>Background Color:</legend><br/> 

<label for="b_r">R:</label> 

<input type="text" id="b r" name="b r" size="3" maxlength="3" 
<label for="b_ g">G:</label> 

<input type="text" id="b g" name="b g" size="3" maxlength="3" 
<label for="b_b">B:</label> 

<input type="text" id="b b" name="b b" size="3" maxlength="3" 
</fieldset> 

<fieldset> 

<legend>Text Color:</legend><br/> 

<label for="t_r">R:</label> 

<input type="text" id="t_r" name="t_r" size="3" maxlength="3" 
<label for="t_g">G:</label> 

<input type="text" id="t_g" name="t_g" size="3" maxlength="3" 
<label for="t_b">B:</label> 

<input type="text" id="t_b" name="t_b" size="3" maxlength="3" 
</fieldset> 


<p><label for="String">Text String:</label> 
<input type="text" id="string” name= String size="35" 


<p><label for="font_size">Font Size:</label> 


<select id="font_size" name="font_size"> 
<option value="1">1</option> 
<option value="2">2</aption> 
<option value="3">3</option> 


<option value="4">4</option> 


{></p> 


{> 


{> 


{> 


/> 


{> 


{> 


98: <option value="5">5</oaption> 

59: </select></p> 

61: <fieldset> 

62: <legend>Text Starting Position:</legend><br/> 

63: <label for="x">X:</label> 

64: <input type="text" id="x" name="x" size="3" maxlength="3" /> 
65: <label for="y">Y:</label> 

66: <input type="text’ id="y" name="y" size="3" maxlength="3" /> 
67: </fieldset> 

69: <button type="submit" name="submit" value="create">Create Image</button> 
70: </form> 

71: </body> 

72: </html> 


这 段 代 码 只 是 一 个 HTML 表 单 ， 它 定义 了 几 个 字段 ， 用 来 获取 图 像 
的 规格 。 可 以 看 到 ， 只 有 第 1 行 到 第 4 行 是 PHP 代 码 ， 并 且 ， 这 些 代 码 行 
只 查看 表单 是 合 已 经 提交 了 《如 果 $_POST 超 全 局 存在 的 话 ) 。 在 查看 
之 后 ， 我 们 跳出 了 PHP， 并 且 为 表单 提供 了 代码 ， 只 有 表单 还 没有 提交 
的 时 候 ， 才 会 显示 它 ; 你 可 以 将 这 段 HTML 放 入 到 PHP 代 码 中 的 一 条 
echo 语 名 中， 但 是 ， 没 有 理由 让 PHP 编 译 器 来 做 这 些 工 作 。 第 24 行 到 第 
26 行 中 的 表单 字段 定义 了 和 想 要 绘制 的 图 像 的 宽度 和 高 度 。 接 下 来 ， 这 上 段 
代码 设 置 了 用 来 为 背景 闫 色 (第 32 行 、 第 34 行 和 第 36 行 ) MERENI 
取 RGB 值 的 几 个 字段 〈 第 42 行 、 第 44 行 和 第 46 行 ) 。 


提示 : 





我 们 可 以 创建 一 个 包含 值 0 到 255 的 下 拉 列 表 框 ， 从 而 获取 红色 、 绿 色 和 音色 的 值 。 这 将 
确 你 用 户 输入 在 要 求 的 范围 之 内 。 





第 50 行 包含 了 一 个 用 来 输入 字符 串 的 表 千 了 学 段 。 这 个 子 人 符 串 将 会 
以 指定 的 文本 闫 色 绘 制 到 图 像 的 背景 之 上 。 第 53 行 到 第 59 行 表示 一 个 用 


来 选择 字体 大 小 的 下 拉 列 表 。 有 5 种 大 小 ， 从 1 到 5， 分 别 表 示 款 认 的 回 
定 宽度 字体 。 用 来 根据 一 个 字符 绅 绘 制 文本 的 玫 数 ， 使 用 5 种 大 小 ， 从 1 
PSR BAR, FARA) ， 用 于 表示 服务 袁 上 安 冯 的 默认 的 回 
FE Di SEEM 


提示 : 


我 们 可 以 使 用 imageloadfont() 和 imagettftextO) 函 数 来 指定 字体 。 详 见 http:/www.php. 


net/image 。 


最 后 ， 第 64 行 和 第 66 行 定义 文本 的 开始 位 置 。 图 像 区 域 的 左上 角 的 
X 位 置 为 0，Y 位 置 也 为 0， 向 下 增加 10 个 像素 的 Y 位 置 为 10， 向 右 增 加 10 
个 像素 的 X 位 置 为 10， 以 此 类 推 。 


如 果 在 这 里 终止 了 脚本 并 且 结 束 了 if...else 语 名 和 PHP 代 码 块 ， 当 把 
这 个 脚本 和 载 入 到 Web 浏 贤 絮 的 时 候 ， 我 们 将 会 看 到 如 图 14-8 所 示 的 一 个 
表单 。 


E Image Creation Form 


€ > C |O http://localhost/14/imagestring.php Kl KK ®& 


Create an Image 
Image Size: 

W: H: 
Background Color: 

及 : G: B: 
Text Color: 


R: G: B: 


Text String: 

Font Size: 1|~| 

Text Starting Position: 
X: Y: 


| Create Image 


图 14-8 ”用户 输入 表单 来 创建 图 像 


只 有 在 第 23 行 以 后 ， 我 们 可 以 完成 这 个 脚本 并 且 产 生 带 有 文本 字 
符 串 的 图 像 ， 因 此 ， 看 看 程序 清单 14.7 其 余 的 内 容 。 


程序 清单 14.7 ”从 用 户 和 输入 创建 一 个 图 像 〈 续 ) 


Jon <7php 
74: } else { 


75: /fcreate image 

TG: //create the canvas 

Pin $myImage = ImageCreate($ POST['w'], $_POST[‘'h']}; 

78 

fo. //set up some colors 

80: $background = ImageColorAllocate ($myImage, $ POST['b r'], 
81: $_POST[ 'b_g'], $_POST['b_b']); 

82: $text = ImageColorAllocate ($myImage, $ POST['t r'], 
83: $ POST['t_g'], $ POST['t_b']); 

84: 

BD // write the string at the top left 

86: ImageString($myImage, $ POST[ font size'], $ POST[‘x'], 
87: $ POST['y'], $_POST['string'], $text); 

88: 

89: / ¿output the image to the browser 

90: header ("Content-type: image/png"); 

91: ImagePNG($SmyImage) ; 

92: 

93: {/f/clean up after yourself 

94: ImageDestroy ($myImage) ; 

95: } 

96: ?> 


第 73 行 到 第 96 行 的 主要 部 分 我 们 之 前 已 经 见 到 过 ， 只 是 这 一 次 我 们 
使 用 从 超 全 局 变量 $_ POST 中 提取 的 元 系 来 取代 直接 编码 的 值 。 在 第 77 
行 ， 我 们 使 用 来 上 自 表 单 的 宽度 值 和 高 度 值 设置 了 初始 图 像 。 第 80 行 到 第 
83 行 定义 了 两 种 颜色 变量 ，$backgroun 和 $text， 它 们 使 用 表单 所 提供 的 
相应 的 RGB 值 。 


提示 : 





在 脚本 中 ， 这 些 颜 色 将 不 会 给 予 实 际 的 颜色 名 ， 因 为 我 们 不 知道 用 记得 入 将 会 产生 什么 
颜色 ， 我 们 可 能 把 这 种 颜色 叫做 $ed。 但 是 如 条 用 户 将 其 定义 为 0，255，0， 我 们 束 会 傻眼 
了 ， 因 为 这 个 RGB 值 表示 绿色 ! 因此 ， 我 们 只 是 根据 闫 色 的 目的 来 命名 闫 色 ， 而 不 是 其 外 
观 。 


第 86 行 到 第 87 行 显示 了 这 个 脚本 中 的 唯一 的 新 项 目 ， 即 


imagestring() PAH EFA. IX SEALE TS BE A (Rit($mylmage). F 
VEK/)\($_POST[ 'font_size' ])、 开 始点 的 X 位 置 和 Y 位 置 ($_POST[ 'x ' ] #1 
$_POST['y '])、 要 绘制 的 字符 串 ($_POSTI[ 'string' ])， 以 及 绘制 字符 串 的 
闫 色 ($text)。 第 90 行 到 第 91 行 把 图 像 输出 到 浏 贤 硕 ， 而 且 第 94 行 销 贤 并 
清除 图 像 创建 过 程 。 


如 果 我 们 把 这 个 文件 保存 为 imagecreate.php， 将 其 放置 到 Web 服 务 
侣 的 文档 根 目 录 下 ， 并 用 填写 表单 ， 输 出 将 会 如 图 14-9 所 示 。 但 是 ， 你 
的 结果 很 可 能 有 上 所 不 同 ， 因 为 有 很 多 变量 起 作用 。 





© imagestring.php (200x50) 


= C © http://localhost/14/imagestring.php Ki A ww 


I like cheese! 


图 14-9 ”图 像 创建 表单 的 示例 输出 


用 各 种 大 小 、 闫 色 和 文本 字符 串 目 己 实 试 一 下 。 


14.7 (EH AREE y A 


前 面 所 有 的 脚本 产生 图 像 输出 ， 但 我 们 把 它们 当 作 单独 的 脚本 调 
用 。 要 在 目 己 的 HIML 中 使 用 脚本 创建 的 图 像 的 结束 ， 这 很 简单 ， 只 需 
要 在 img 标签 的 sre 属性 中 使 用 脚本 而 个 是 图 像 》 的 名 字 ， 示 例如 
Fe 


<img src="NameOfScript.php" /> 


EETA F148, ERAT sae Sd AE EY BAS OR E — I VR 
Mia, Himig ve ix SAR, FHEA bias FANE o 


为 了 添加 一 些 新 的 功能 ， 这 个 基本 脚本 载 入 并 使 用 一 种 定制 字体 来 
把 文本 字符 串 显 示 为 一 个 图 形 ， 和 前 面 小 节 中 用 户 生 成 的 文本 很 相似 ， 
只 不 过 字体 有 趣 一 些 。 这 么 做 的 原因 将 很 快 会 消 芍 。 站 完 ， 创 建生 成 图 
像 的 脚本 。 程 序 清单 14.8 给 出 了 一 个 例子 。 





程序 清单 14.8 ”使 用 定制 字体 和 文本 创建 一 个 图 像 


<?php 
//create the canvas 
$myImage = ImageCreate(150, 25) ; 


/{/set up some colors for use on the canvas 
$white = ImageColorAllocate($myImage, 255, 255, 255); 
$black = ImageColorAllocate($myImage, @, ©, @); 


ON oar Wh 一 


< 


/{load a font 
10: $font = imageloadfont("hootie.gdf"); 


12: // write the string 
13: ImageString($myImage, $font, @, ©, "CAPTCHA!", $black); 


15: /f/output the image to the browser 
16: header ("Content-type: image/png"); 
17: ImagePng($myImage) ; 


19: //clean up after yourself 
20: ImageDestroy({$myImage) ; 
?> 


第 3 行 创 建 了 一 个 宽 150 像 素 高 25 像 素 的 画布 。 这 个 尺寸 和 我 们 通常 
在 CAPTCHA 中 见 到 的 内 容 大 小 相同 。 实 际 上 ， 这 就 是 我 们 所 创建 的 内 
容 的 一 部 分 。 


提示 : 


CAPTCHA 是 一 个 图 形 化 的 挑战 -响应 测试 ， 用 来 确定 用 户 是 否 是 人 类 和 而 非 机 器 。 你 可 
能 会 在 下 列 情况 下 碰 到 过 CAPTCHA: 在 完成 一 个 表单 ， 从 而 在 站 点 上 留 下 一 条 评论 或 参与 一 
个 讨论 论坛 的 时 候 ， 或 者 在 创建 一 个 用 户 账户 的 时 候 。 其 思想 是 ， 只 有 人 能 够 阅读 图 像 中 的 
字符 文本 。 我 们 可 以 从 http://en.wikipedia.org/wiki/Captcha 了 解 有 关 CAPTCHA 的 更 多 内 容 。 





BOTT PAT AZHAR RE Se: 日 色 作 为 背景 闫 色 和 黑色 作为 
男 一 种 颜色 。 第 10 行 使 用 imageloadfont() 函 数 载 入 字体 。 如 果 想 要 使 用 
TrueType 字 体 ， 也 可 以 使 用 imagettftextO 函 数 。 在 这 个 例子 中 ， 
hootie.gdf 字 体 是 可 以 目 由 获取 的 字体 ， 它 取 目 于 文件 系统 中 的 一 个 单个 


PKI 
PKI 


文件 ， 并 且 由 调用 它 的 函数 载 入 。 


提示 : 


我 们 可 以 通过 在 Google 或 者 其 他 搜索 引擎 中 搜索 “free GDF font”* 或 “free TrueType font”, 
以 找到 这 个 例子 中 使 用 的 字体 ， 以 及 其 他 的 几 种 免费 字体 。 





第 13 行 使 用 第 10 行 载 入 的 字体 作为 InageStringO 函 数 调 用 的 一 部 
分 。 在 这 个 例子 中 ，$font 是 第 二 个 参数 ， 意 味 看 包含 在 字体 包 中 的 字体 
和 字体 大 小 用 于 这 个 函数 的 实例 中 。 开 始 的 X 和 YY 位 置 分 别 在 第 3 个 
参数 和 第 4 个 参数 中 和 直接 编码 为 0， 并 且 将 要 绘制 的 字 从 串 CAPTCHA! 
也 是 直接 编码 的 。 绘 制 这 个 字符 串 的 磊 色 是 $black， 这 是 最 后 一 个 参数 
的 值 。 


第 16~~17 行 把 图 像 数 据 沈 输 出 到 Web 浏 哆 器 : 站 先 发 送 相 应 的 
header() 函 数 ， 然 后 使 用 ImagePng0 输 出 数据 流 ， 这 个 例子 输出 了 一 个 
PNG 文 件 。 第 20 行 使 用 ImageDestroy0 函 数 清 除了 脚本 开始 处 的 
ImageCreate() 函 数 所 使 用 的 内 存 。 


如 果 在 此 时 仿 下 来 ， 把 这 个 脚本 友 运 到 Web ARS HIF Be TE WM A 
器 中 载 入 它 ， 将 会 以 定制 字体 显示 “CAPTCHA!”。 但 是 ， 本 节 的 目的 是 
展示 如 何 使 用 img 标 等 来 载 入 图 像 ， 以 及 展示 需要 一 些 HIML 文 件 来 完 
成 什么 。 程 序 清单 14.9 包 含 了 完成 这 项 工作 的 一 个 img 标 签 。 





程序 清单 14.9 ”使 用 定制 字体 和 文本 创建 一 个 图 像 


<!DOCTYPE html> 

<html> 

<head> 

<title>Using Images Created by Scripts</title> 
</head> 


ab G N 一 


6: </body> 

7: <hi>Generated Image Below...</hi> 

8: <img src="imagecustomfont.php" alt="generated image" /> 
9: </body> 

10: </html> 


这 个 人 简短 的 HTML 文 件 包 含 了 有 趣 的 一 行 ， 即 第 8 行 。 在 第 8 行 ， 我 
们 可 以 看 到 img 标 签 的 sre 属性 的 值 是 程序 清单 14.8 所 创建 的 脚本 的 名 
=, BẸ imagecustomfont.php。 把 这 个 HTML 文件 取 名 为 useimage.html， 
FFSEHTMLA#PHP HAAS BB ACIS 25 Web ERS Air o 


在 浏览 右 中 打开 useimage.html， 你 将 会 看 到 如 图 14-10 所 示 的 内 容 ， 
即 以 定制 字体 显示 的 “CAPTCHA!”， 这 实际 上 是 以 一 个 图 形 显 示 的 。 


| Using Images Created by 5 
€ C | © hitp://localhost/14/useimage.htm Ki E 
Generated Image Below... 


CAPTCHA! 


图 14-10 ”使 用 脚本 创建 的 图 像 


如 果 你 真 地 使 用 这 个 脚本 作为 CAPTCHA 系 统 的 一 部 分 ， 至 少 要 对 
其 做 如 下 两 方面 的 完善 。 随 机 化 字符 串 ， 而 不 是 对 其 直接 编码 ;把 随机 
产生 的 字符 串 保 存在 一 个 数据 库 表 中 ， 以 便 能 够 进行 输入 匹配 过 程 。 


14.8 24 


本 章 针 对 我 们 能 够 使 用 图 像 和 PHP 做 什么 给 出 了 一 个 简短 的 介 

。 我 们 强烈 推荐 位 于 http://www.php.net/image 的 PHP 手 册 的 “Image 
Functions” 部 分 ， 不 仅 因 为 这 是 和 图 像 相 关 的 函数 的 一 个 完整 列表 ， 而 
日 这 里 还 有 很 多 有 天 它 们 在 应 用 程序 中 的 用 法 的 例子 和 讨论 。 


在 本 章 中 ， 我 们 学 习 了 安 寂 和 使 用 附加 的 库 来 操作 图 像 ， 以 PNG 
为 例 ， 但 是 ， 我 们 也 给 出 了 使 用 GIF 和 JPG 的 说 明 。 我 们 学 习 了 创建 一 
个 图 像 男 布 和 分 配 颜 色 的 基本 步骤 ， 以 及 绘 Wn 
我 们 学 习 了 绘制 男 布 可 以 由 一 个 已 有 的 图 像 组 成 ， 还 学 习 了 合并 层 以 使 
最 终 图 像 是 一 个 合并 了 层 的 合成 图 像 。 帮 外， 我 们 看 到 了 在 图 像 创建 肢 
本 中 《在 例子 中 是 从 一 个 HTML RE) 使 用 用 户 输 入 是 多 么 简单 ， 以 
及 如 何在 HTML 代码 中 把 脚本 作为 图 像 源 使 用 。 


14.9 Q&A 


Q: 如 何 使 用 动态 数据 来 创建 一 个 分 基 的 饼 图 ? 


A: 创建 任何 图 形 的 时 候 ， 开 始点 和 绘制 长 度 都 不 需要 静态 指定 ， 
它们 可 以 是 变量 ， 其 值 由 数据 库 、 用 户 输入 或 者 脚本 内 的 计算 所 确定 。 
人 例如， 下面 这 段 代 码 创 建 一 个 红色 的 实心 90 度 弧 形 。 


ImageFilledArc ($myImage ,50,50,100,50,0,98,$red,IMG ARC PIE); 

我 们 可 以 这 样 设置 ， 使 得 位 于 人 饼 图 顶部 的 红色 填 元 的 弧 形 把 May 
Sales 占 总 数 的 百分比 保存 到 一 个 名 为 $may_sales_pct 的 变量 中 。 然 后 ， 
这 行 代 人 码 变 为 如 下 所 示 。 
ImageFilledArc($myImage,58,50,108,50,0,$may sales pct,$red,IMG ARC PIE}; 


BRE, BRAES AAE P A EAE E A A ABR HT 
BAS ANAS RS WE A IE I AM 360E. 


14.10 ”实践 练习 


这 个 实践 练习 设计 用 来 帮助 你 预测 可 能 的 问题 、 复 习 已 经 学 过 的 知 
识 ， 并 且 开 始 把 知识 用 于 实践 。 


问答 题 
1. 表示 纯 黑 色 和 纯 白 色 的 RGB 值 古 多 少 ? 
2. 如 何 创建 一 个 新 的 、 空 白 的 、300 像 素 宽 、200 像 素 高 的 画布 ? 


3， 哪 个 函数 用 来 绘制 多 边 形 ? 绘制 一 个 填充 的 多 变形 呢 ? 


1. 0, 0, 0) 是 纯 黑 色 ， (255, 255, 255) 是 纯 白 色 。 


2. 使 用 如 下 命令 ， 创 建 一 个 新 的 、 黑 色 的 画布 ， 宽 300 像 素 ， 高 
200 像 素 。 


$new image = ImageCreate(300, 200}; 


3. ImagePolygon0 和 ImageFilledPolygon()。 


EA wel 


1. TEAM HE BE TARA ROA, KARR A E BOK 
平 的 条 块 ， 每 个 IRRE. FFARR Needs, (EAA AA 
的 长 度 。 确 保 每 个 条 块 之 间 有 10 个 像 系 的 空间 。 


2. 扩展 思考 题 1 NG, TESST, IVE PA 
目 己 选择 的 数值 。 根 据 这 些 数值 来 生成 条 状 图 。 


haw 
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第 4 部 分 “PHP 与 MySQL 整 合 


理解 数据 库 设 计 过 程 
SQL 基本 命令 
使 用 MySQL 中 的 事务 和 存储 过 程 


使 用 PHP 和 MySQL 交互 


第 15 章 ”理解 数据 库 议 计 过 程 


本 章 所 小 及 的 主题 包括 : 


。 民 好 的 数据 库 设 计 的 一 些 优 操 。 

。 表 关系 的 三 种 类 型 。 

。 如 何 规范 化 数据 库 。 

。 如 何 实现 一 个 展 好 的 数据 库 设 计 过 程 。 


人 在 本 章 中 ， 我 们 将 学 习 设 计 一 个 关系 数据 库 痛 后 的 过 程 。 在 这 个 关 
注 理论 的 一 章 中 ， 我 们 直接 开始 学 习 基 本 的 MYSQL 命令 ， 为 把 MySQL 
整合 到 目 己 的 应 用 程序 中 而 做 准备 。 


15.1 民 好 的 数据 库 议 计 的 重要 性 


民 好 的 数据 库 设 计 对 于 一 个 高 性 能 的 应 用 程序 非 帝 重要 ， 驳 像 一 个 
空气 动力 农 置 对 于 一 辆 餐车 的 备 要 性 一 样 。 如 果 一 辆 汽车 没有 平滑 的 曲 
线 ， 将 会 产生 阻力 从 而 变 慢 。 关 系 没有 经 过 优化 ， 数 据 库 无 法 尽 可 能 局 
效 地 运行 。 应 该 把 数据 库 的 关系 和 性 能 《包括 易 维护 性 、 最 小 化 、 可 重 
复 性 以 及 避免 不 一 致 性 ) 看 作 是 规范 化 的 一 部 分 。 


提示 : 














规范 化 指 的 是 为 了 尽量 避免 重复 性 和 不 一 致 性 而 组 织 数据 结构 的 过 程 。 


除了 性 能 以 外 的 问题 ， 束 是 维护 的 问题 了 ， 数 据 库 应 该 易于 维护 。 
ELTA BCE AREY CORA GE) 重复 性 数据 。 如 采 有 很 多 的 重 
复 性 数据 ， 这 些 数 据 的 一 个 实例 及 生 一 次 改变 《〈 例 如， 一 个 名 字 的 改 
变 ) ， 这 个 改变 必须 对 所 有 的 其 他 的 数据 都 进行 。 为 了 避免 重复 ， 并 且 
增强 维护 数据 的 能 力 ， 我 们 可 以 创建 可 能 的 值 的 一 个 表 并 使 用 一 个 键 来 
引用 该 值 。 在 这 种 方式 中 ， 如 来 值 改变 了 了 名字， 这 个 改变 只 在 主 表 中 友 
生 一 次 ， 所 有 的 其 他 表 的 引用 都 保持 不 变 。 


例如 ， 假 设 你 负 贡 维护 一 个 和 学生 数据 库 以 及 他 们 所 注册 的 课程 。 如 
果 有 35 个 学 生 在 同一 个 课 党 中 ， 让 我 们 将 这 门 课 叫 做 Advanced 
Math (EFA) ， 谍 程 的 名 字 将 会 在 表 中 出 现 35 次 。 现 在 ， 如 采 老 师 
决定 把 这 门 课 的 名 字 改 为 Mathematics IV， 我 们 必须 修改 35 条 记录 以 反 
映 出 新 的 诛 程 名 。 如 末 数 据 库 设 计 为 诛 程 名 出 现在 一 个 表 中 ， 只 有 读 程 


IDS tA FEWR eR, AS A RE 4, EA at Re BE 
一 条 记录 而 不 是 35 条 记录 。 


一 个 规划 和 设计 民 好 的 数据 库 的 优点 是 众多 的 ， 它 也 证 实 了 这 样 一 
个 道理 ， 前 期 做 的 工作 越 多 ， 后 面 所 要 做 的 束 越 少 。 在 使 用 数据 库 的 应 
用 程序 公开 及 布 之 后 ， 还 要 对 数据 库 进 行 重 新 设计 ， 这 十 最 粮 糙 的 ， 然 
mM. AMESZ, HRI E r o 

因此 ， 在 开始 编 与 一 个 应 用 程序 的 代码 之 前 ， 请 化 大 量 的 时 间 来 设 


计 你 的 数据 库 。 在 本 重 其 余 的 部 分 中 ， 我 们 将 学 习 很 多 有 关头 系 和 规范 
化 的 内 容 ， 这 十 设 计 难 题 中 最 重要 的 两 部 分 。 


15.2 a n 


表 天 系 上 其 有 以 下 儿 种 形式 。 


e 一 对 一 关系 。 
© 一 对 多 关系 。 
© 多 对 多 天 系 。 


例如 ， 假 设 有 一 个 名 为 employees 的 表 ， 其 中 包含 了 每 个 人 的 社会 
安全 号 介 、 名 字 和 他 所 在 的 部 门 。 假 设 我 们 还 有 一 个 名 为 departments 的 
表 ， 其 中 包 售 了 所 有 部 门 的 列表 ， 每 个 部 门 有 一 个 部 门 ID 和 一 个 和 名字。 
在 emplovyees 表 中 ， 部 门 ID 字段 和 departments 表 中 的 ID 字段 相对 应 。 我 
们 可 以 在 图 15-1 中 见 到 这 种 类型 的 关系。 字段 名 劳 边 的 PK 表示 该 表 的 主 
BE 


employees 


(PK) SS Number 
DeptlD 


FirstName 
departments LastName 


@ DeptiD 


DeptName 










图 15-1 employees 表 和 departments 表 通过 DeptID 键 联系 起 来 
在 下 面 的 小 节 中 ， 我 们 将 进一步 详细 介绍 每 种 关系 。 


15.2.1 一 对 一 关系 


在 一 对 一 的 关系 中 ， 一 个 键 只 能 在 一 个 关系 表 中 出 现 一 次 。 
employees 表 和 departments 表 没有 一 对 一 的 关系 ， 因 为 显然 很 多 职员 都 属 
于 同一 个 部 门 。 但 是 ， 如 末 公 司 中 的 每 个 职员 分 配 了 一 台 计 算 机 ， 这 吏 
存在 一 对 一 的 关系 了 。 图 15-2 给 出 了 职员 和 计算 机 之 间 的 一 对 一 的 天 


EMPLOYEE 1 COMPUTER 1 
EMPLOYEE 2 COMPUTER 2 





EMPLOYEE 3 COMPUTER 3 
EMPLOYEE 4 COMPUTER 4 


数据 库 中 的 employees 表 和 computers 表 看 上 去 如 图 15-3 所 示 ， 表 示 
了 它们 一 对 一 HK 3 So 





employees 







(PK) so Number 
DeptlD 
FirstName 
LastName 
Computer!D 






computers 


a ComputerlD 
ComputerDesc 


图 15-3 ”数据 模型 中 的 一 对 一 关系 
15.2.2 一 对 多 关系 


在 一 个 一 对 多 关系 中 ， 一 个 表 中 的 键 在 一 个 相关 的 表 中 出 现 多 次 。 
如 图 15-1 中 的 例子 所 示 ， 它 表示 的 是 职员 和 部 门 之 间 的 关系 ， 即 一 个 一 
对 多 的 关系。 现实 中 的 一 个 例子 就 古 部 门 的 组 织 图 ， 如 图 15-4 所 示 。 


DEPARTMENT 
A 


EMPLOYEE EMPLOYEE 2 EMPLOYEE 3 EMPLOYEE 4 


图 15-4 一 个 部 门 包 含 了 多 个 职员 


一 对 多 的 关系 是 最 为 弟 见 的 一 种 天 系 。 男 一 个 实际 的 例子 束 是 在 一 
个 地 址 数据 库 中 使 用 一 个 州 的 缩写 ,每 个 州都 有 一 个 唯一 的 标识 答 
(CA 表示 California、PA 表 示 Pennsylvania 等 ) ， 并 且 美 国 的 每 个 地 址 都 
有 一 个 相关 的 州 。 


如 果 我 们 有 8 个 朋友 在 California 并 有 日 有 5 个 朋友 在 Pennsylvania， 在 
BATH eR A EPPS AS RS CCA) 表示 一 个 一 对 
八 的 关系 ， 夯 一 个 缩写 PA) 表示 一 个 一 对 五 的 关系 。 


15.2.3 ”多 对 多 关系 
多 对 多 关系 通 负 会 在 规范 化 的 数据 库 的 实际 例子 中 引发 问题 ， 以 至 


Pill i BRIE SX RAIA BI OY BRR LEB BRA 
H, SRN BEE AT DAE I eK Al, “Er EA 
像 旦 一 个 一 对 多 关系 ; BERLINA, METRE NERE 
个 表 中 出 现 多 次 。 


按照 这 样 一 种 方式 来 思考 一 种 关系 ， 可 以 以 学 生 和 读 程 为 例 : 一 个 
学 生 有 一 个 ID 和 一 个 名 罕 ; 一 门 读 程 有 一 个 ID 和 一 个 名 衬 。 正 如 图 15-5 
所 示 ， 一 个 学 生 可 能 一 次 选 多 门 读 程 ， 而 一 门 读 程 总 是 有 一 个 以 上 的 学 
aor 


STUDENT 1 SI VEEN 
STUDENT 2 STUDENT 6 
CLASS 
A 
STUDENT 3 
STUDENT 7 
STUDENT 4 CLASS 


C 





CLASS 


图 15-5 FERIERE WHEUA FE 


正如 你 所 看 到 的 ， 这 种 关系 不 能 以 一 种 相关 表 的 简单 方法 表示 。 这 
个 表 也 可 能 看 上 去 如 图 15-6 所 示 ， 似 乎 是 不 相关 的 。 


students classes 


@ StudentiD @ ClassiD 


FirstName ClassDesc 
LastName 





图 15-6 学生 表 和 班级 表 不 相关 


为 了 让 多 对 多 关系 更 理论 化 ， 我 们 可 能 创建 一 个 中 间 表 ， 这 个 表 位 
于 两 个 表 之 间 ， 并 且 实 际 上 把 它们 映射 到 一 起 。 我 们 可 以 构建 和 图 15-7 
中 所 示 的 表 类 似 的 一 个 表 。 


students_classes_map 


students 
classes 


(PK) StudentID 
FirstName . a9 ClassiD 
LastName ClassDesc 





415-7 students_classes_map 表 充当 一 个 中 间 表 


如 果 取 出 图 15-5 中 的 信息 并 且 将 其 放 入 到 这 个 中 间 表 中 ， 我 们 会 得 
到 如 图 15-8 所 示 的 结果 。 


图 15-8 填充 了 数据 的 students_classes_map 表 





可 以 看 到 ， 多 个 学 生 和 多 门 课程 在 students_classes_map 表 中 
愉快 地 共存 。 


在 介绍 了 关系 的 闫 型 之后， 我们 再 来 学 习 规 范 化 融 是 小 荣 一 代 了 。 


15.3 理解 规范 化 


规范 化 只 是 一 组 规则 ， 当 我 们 作为 一 名 数据 库 管 理 员 的 时 候 ， 这 矢 
规则 会 让 我 们 的 生活 很 容易 。 投 照 这 样 一 种 方式 来 组 织 数 据 库 ， 使 得 其 
相关 的 表 能 够 适应 和 元 活 应 对 未 来 的 增长 ， 这 二 一 门 技 艺 。 


规范 化 中 用 到 的 这 一 套 规则 叫做 范式 。 如 果 数据 库 设 计 遵从 了 第 一 
组 规则 ， 它 就 考虑 使 用 第 一 范式 。 如 果 遵 从 了 前 3 组 规则 ， 就 说 明 我 们 
的 数据 库 使 用 了 第 三 范式 。 


在 整个 本 章 中 ， 我 们 将 学 习 第 一 和 范式、 第 二 范式 和 第 三 范式 中 的 每 
条 规则 。 我 们 希望， 在 你 创建 自己 的 应 用 程序 的 时 候 能 够 遵从 它们 。 下 
面 还 将 使 用 学 生 和 课程 数据 库 的 一 组 示例 表 ， 并 且 用 第 三 范式 规范 它 
们 ]。 


15.3.1 平 表 市 来 的 问题 


在 开始 学 习 第 一 范式 之 前 ， 我 们 必须 从 需要 修复 的 地 方 开始 。 在 一 
个 数据 库 的 例子 中 ， 这 束 是 一 个 平 示 (flat table) - FRIZE DK 
单 ， 它 有 很 多 很 多 的 列 。 多 个 表 之 间 没 有 关系 ， 我 们 需要 的 所 有 数据 都 
可 能 在 这 个 平 表 之 中 。 这 是 一 种 没有 效 京 的 情景 ， 而 且 会 比 规 范 化 的 数 
据 库 消耗 挥 更 多 的 便 盘 物理 空间 。 


在 学 生 和 读 程 数据 库 中 ， 假 设 在 平 表 中 有 如 下 的 字段 。 


PEE 44 Fo 
。 CourseID1 —— F Æ 10 hE — REID. 


e StudentName 





CourseDescription1 一 一 学 生 选 择 的 第 一 门 诬 程 的 说 明 。 
学 生 选 择 的 第 一 门 诛 程 的 老师 。 
CourseID2 一 一 学 生 选 择 的 第 二 门 读 程 的 ID。 
CourseDescription2 一 一 学 生 选 择 的 第 二 门 诬 程 的 说 明 。 
学 生 选 择 的 第 二 门 课 程 的 老师 。 
针对 学 生 在 他 们 的 学 业 中 涉及 的 所 有 诛 程 ，CourseID、 
CourseDescription 和 CourseInstructor 列 可 以 多 次 重复 出 现 。 


CourselInstructor1 





CourselInstructor2 





根据 我 们 目前 已 经 学 习 的 内 容 ， 我 们 应 该 能 够 识别 第 一 个 问题 : 
CourseID 、CourseDescription 和 CourseInstructor 列 是 重复 的 组 。 


去 除 元 余 是 规范 化 的 第 一 步 ， 因 此 ， 接 下 来 我 们 要 把 平 表 纳入 到 第 


一 范式 中 。 如 果 我 们 的 表 保留 了 平 的 格式 ， 我 们 可 能 有 很 多 未 占用 的 空 
间 并 且 很 多 空间 都 没 必要 使 用 ， 这 不 是 一 个 有 效率 的 表 设计 。 


15.3.2 ”第 一 苑 式 
第 一 范式 的 规则 如 下 。 


。 去 除 重 复 的 信息 。 
© 为 相关 的 数据 单独 创建 一 个 表 。 


如 果 考 虑 学 生 和 课程 数据 库 使 用 市 有 很 多 午 复 的 字段 组 的 平 表 设 
计 ， 我 们 可 以 区 分 两 个 明显 的 主题 ， 即 学 生 和 课程 。 把 学 生 和 课程 数据 
库 纳 入 到 第 一 范 芭 ， 意 味 独 我 们 要 创建 两 个 表 ， 一 个 用 于 学 生 ， 一 个 用 
于 诛 程 ， 如 图 15-9 所 示 。 


students 


@ StudentiD 
StudentName 


student courses 


StudentID 
CourselD 
CourseDescription 
Courselnstructor 





图 15-9 ”把 平 表 分 解 为 两 个 表 


两 个 表现 在 表示 了 一 个 学 生 到 多 门 诛 程 的 一 对 多 的 天 系 。 学 生 可 以 
选取 尽 可 能 多 的 课程 ， 而 不 会 受到 平 表 中 己 有 的 
CourseID/CourseDescriptiomn/CourseInstructor 组 的 数目 的 限制 。 


一 步 把 这 些 表 纳 入 第 二 汇 式 。 
15.3.3 ”第 二 范式 
第 二 范 陈 的 规则 如 下 : 
。 没有 依赖 于 主键 的 真子 集 的 非 主 键 属性 。 


用 通俗 易 懂 的 语言 来 说 ， 这 意味 着 ， 如 果 表 中 的 字段 不 是 完全 和 一 
个 主键 相关 ， 我 们 有 更 多 的 工作 要 人 做。 在 学 生 和 课程 的 例子 中 ， 我 们 需 
要 把 诛 程 分 解 到 他 们 目 己 的 表 中 并 且 修 改 students_courses 表 。 


CourseID 、CourseDescription 和 CourseInstructor 可 能 成 为 一 个 叫做 
courses 的 表 ， 它 拥有 一 个 CourseID 主 键 。 随 后 ，students_courses 表 应 该 
只 包含 两 个 字段 StudentID 和 CourseID。 我 们 可 以 在 图 15-10 中 看 到 这 个 
新 的 设计 。 


students 
courses 


@ StudentiD - e 
StudentName PK Class|D 


student_courses ClassDescription 
StudentID Courselnstructor 
Class|ID 





415-10 ”使 表 符 合 第 二 范式 
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15.3.4 ”第 三 范式 
第 三 苑 了 式 的 规则 如 下 所 示 。 
e。 没有 依赖 于 非 主 键 属性 的 属性 。 


这 条 规则 只 是 意味 看 我 们 需要 和 奉 看 表 ， 看 看 表 是 否 有 更 多 的 字段 可 
以 进一步 分 解 ， 以 便 可 以 不 再 依赖 于 一 个 键 。 考 处 删 际 重复 的 数据 ， 我 
们 将 找到 答案 ， 这 就 古 教师 。 不 可 避免 的 ， 一 个 教师 将 要 教授 多 | 门 课 
程 。 然 而 ，CourseInstructor 个 是 任何 类 型 的 一 个 主键 。 因 此 ， 如 果 我 们 
把 这 个 信息 分 解 并 且 纯 粹 为 了 效率 和 可 维护 性 来 创建 一 个 单独 的 表 〈 如 
图 15-11 所 示 ) ， 这 就 是 第 三 范式 。 


courses 


my ClassiD 
ClassDescription 


CourselnstructorID 
students 


instructors 


@ StudentiD 


StudentName m m9 InstructoriD 
student courses 


InstructorName 
StudentlD InstructorNotes 


CourselD 





图 15-11 把 表 纳 入 第 三 范式 


第 三 范式 通 钊 删除 了 利 见 的 见 余 并 且 考 碟 到 了 元 活性 和 增长 性 。 下 
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用 程序 的 全 面 设计 过 程 中 给 出 一 些 指南 。 


15.4 Mirit 
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驱动 的 应 用 程序 ， 设 计 过 程 必 须 包 仿 对 数据 的 全 面 考虑 ， 这 应 当 包 括 数 
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设计 过 程 中 的 一 般 步 又 如 下 。 


定义 目标 。 

WI BEAN CR. FBO) 。 
分 清关 系 。 
定义 和 实现 业务 规则 。 
创建 应 用 程序 。 


创建 应 用 程序 是 最 后 一 步 ， 而 不 是 第 一 步 。 很 多 开发 者 从 应 用 程序 
开始 构建 它 ， 然 后 回 过 头 来 试图 构建 一 组 数据 库 表 来 填 入 应 用 程序 的 数 
据 。 这 种 方法 完全 是 南 辕 北 略 ， 效 率 低下 ， 并 且 花 费 很 多 的 时 间 和 金 
oe 


在 我 们 开始 任何 应 用 程序 设计 过 程 之 前 ， 坐 下 来 进行 详细 的 讨论 。 
如 于 我 们 不 能 摘 述 应 用 程序 ， 包 括 目 标 、 用 户 和 目标 市 场 ， 那 么 ， 我 们 
还 没有 准备 好 构建 它 ， 更 不 要 说 对 数据 库 建 模 了 。 


在 可 以 同 其 他 人 描述 应 用 程序 的 功能 和 兰 列 ， 并 且 使 这 对 他 们 有 和 意 
Mila, BANAT AFRA ERRE ENKI o FEM PARED PRH 
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也 不 为 过 。 


下 一 步 束 是 进行 规范 化 。 从 平 表 到 第 一 范式 ， 然 后 ， 可 能 的 话 继续 
升级 到 第 三 范式 。 使 用 纸 、 铅 笔 、 记 事 贴 等 任何 能 够 有 助 于 可 视 化 表 和 
天 系 的 工具 。 在 你 准备 目 己 创建 表 之 前 ， 在 记事 贴 上 进行 数据 建 模 ， 这 
没什么 丢人 的 。 为 外 ， 使 用 记事 贴 比 购 天 软件 来 进行 建 柑 要 便宜 很 多 ， 
建 模 软件 的 价格 从 数 百 美元 到 数 干 关 元 不 等。 


在 我 们 有 了 一 个 初步 的 数据 模型 之 后 ， 从 应 用 程序 的 角度 看 看 它 ， 
或 者 从 我 们 所 构建 的 应 用 程序 的 用 户 的 角度 来 看 看 它 。 我 们 也 正 是 从 这 
个 角度 来 定义 业务 规则 并 且 看 看 数据 模型 是 否 违 反 了 规划。 例如 ， 一 个 
在 线 注册 应 用 程序 的 一 条 业务 规则 是 “每 个 用 尸 必须 央 一 个 Email 地 址 ， 
并 且 它 必须 不 属于 任何 其 他 已 有 的 用 户 ?”。 如 末 EmailAddress 在 你 的 数据 
模型 里 不 是 一 个 唯一 的 字段 ， 你 的 模型 将 会 违反 这 一 业务 规则 。 


在 业务 规则 已 经 应 用 到 数据 模型 后 ， 只 有 这 时 候 才 可 以 开始 应 用 程 
序 编程 。 只 要 确保 数据 模型 是 健壮 的 ， 我 们 就 可 以 安心 了 ， 不 需要 把 自 
己 也 加 入 到 程序 编写 中 去 。 后 续 的 事情 没什么 特别 的 。 


15.5 小结 
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系 ， 而 不 是 把 所 有 的 数据 都 一 股 脑 族 到 一 个 长 长 的 平 表 文件 中 。 关 系 的 
类 型 包括 一 对 一 关系 、 一 对 多 关系 和 多 对 多 关系 。 


使 用 关系 来 恰当 地 组 织 数据 ， 这 叫做 规范 化 。 有 很 多 层级 的 规范 
化 ， 但 是 主要 的 层级 是 第 一 范式 、 第 二 范式 和 第 三 范 式 。 每 个 层级 都 有 
必须 遭 守 的 一 条 或 两 条 规则 。 胆 守 所 有 的 这 些 规则 将 确 你 我 们 的 数据 厅 
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要 让 一 个 思想 从 刚刚 起 步 到 产生 成 有 末 ， 我 们 需要 这 从 一 个 设计 过 
程 。 这 个 过 程 通常 是 “三 思 而 后 行 ”。 讨 论 规 则 、 需 求 以 及 目标 ， 然 后 创 
建 规范 化 表 的 最 后 版 本 。 


15.6 Q&A 


Q: 只 有 3 种 泄 式 ? 


A: Ae. ABM. AN Erte Boyce-Codd 范式 、 第 四 
范式 和 第 五 范式 〈 也 称 为 Join-Projection 范 式 ) 。 在 实际 的 应 用 程序 开发 
中 ， 通 痢 并 不 使 用 这 些 范 式 ， 因 为 坦 守 这 些 范 式 需 要 付出 的 人 力 和 数据 
库 效率 代价 太 大 【但 如 果 你 实现 它们 的 话 ， 有 可 能 很 不 错 〉 。 要 了 解 更 
多 信息 ， 请 访问 


http://en.wikipedia.org/wiki/Database_normalization#Normal_forms 。 


15.7 ”实践 练习 


这 个 实践 练习 设计 用 来 帮助 你 预测 可 能 的 问题 、 复 习 已 经 学 过 的 知 
识 ， 并 且 开 始 把 知识 用 于 实践 。 


问答 题 


1. 说 出 3 种 数据 关系 类 型 。 


2. 由 于 多 对 多 的 关系 在 局 效率 的 数据 库 设 计 中 很 难 表示 ， 我 们 该 
怎么 做 呢 ? 


3， 说 出 创建 数据 关系 可 视 化 的 几 种 方式 。 


oy 


ee 
1， 一 对 一 关系 、 一 对 多 关系 和 多 对 多 关系 。 
2， 使 用 中 间 映 射 表 来 创建 一 系列 一 对 多 的 关系 。 


3. 你 可 以 使 用 各 种 工具 ， 从 记事 赔 和 字符 串 注意， 其 中 是 表格 
以 及 表示 表格 之 间 的 关系 的 字符 串 〉， 到 用 于 绘图 的 软件 ， 到 说 明 SQL 
语句 和 提供 可 视 化 的 软件 程序 。 


EA wel 


问 使 用 表单 和 平 示 的 一 个 人 解释 全 部 3 种 范式 。 


第 16 章 ”SQL 基本 命令 


在 本 章 中 ， 我 们 将 学 习 : 


基本 的 MySQL 数 据 类 型 。 

如 何 使 用 CREATE TABLE 命 令 来 创建 表 。 

如 何 使 用 INSERT 命 令 来 输入 记录 。 

如 何 使 用 SELECT 命令 来 获取 记录 。 

Ou {hI FE SELECT #234 sh 18 FE AS eB. WHERE FFL 
GROUP BY 子 句 。 

如 何 使 用 JOIN 或 子 查 询 从 多 个 表 中 查询 。 

如 何 使 用 UPDATE 和 REPLACE 命 令 来 修改 已 有 记录 。 
如 何 使 用 DELETE 命 令 来 删除 记录 。 

如 何 使 用 MySQEL 内 建 的 字符 串 函 数 。 
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上 一 章 已 经 介绍 了 数据 库 设计 过 程 的 基础 知识 ， 本 章 介 绍 核心 SQL 
语法 的 初步 知识 ， 我 们 将 会 使 用 这 些 语法 来 创建 和 操作 MySQL 数 据 库 
表 。 这 是 动手 实践 的 一 草 ， 并 且 假 设 你 能 够 直接 同 MySQL 发 布 命令 ， 
要 么 通过 MySQL 命 令 行 界面 ， 或 者 通过 phpMyAdmin 这 样 的 管理 界面 。 
在 本 书 第 1 章 的 Quick Start 过 程 的 XAMPP 安 装 中 包含 了 phpMyAdmin。 


请 注意 ， 这 可 能 不 是 本 书 中 最 令 人 激动 的 一 草 ， 但 它 将 展示 很 多 元 
系 的 操作 示例 ， 你 将 在 以 后 的 工作 中 用 到 这 些 元 系 。 


16.1 MySQL 数 据 类 型 


在 表 中 正确 地 定义 字段 对 于 全 面 优化 数据 库 很 重要 。 我 们 应 该 对 子 
段 只 使 用 真正 需要 使 用 的 类 型 和 大 小 ， 如 来 我 们 只 二 要 使 用 两 个 字符， 
束 个 要 把 一 个 字段 定义 为 10 个 字符 的 完 度 ， 数 据 库 必须 考虑 那 8 个 籁 外 
的 学 从 ， 即 便 不 会 使 用 它们 。 这 些 字 段 的 类 型 也 叫做 数据 类 型 ， 因 为 我 
们 将 要 在 这 些 字 段 中 和 存 入 “该 类 型 的 数据 ”。 


MySQL 使 用 多 种 不 同 的 数据 类 型 ， 这 些 数 据 类 型 可 以 划分 为 3 类 : 
数字 类 型 、 日 期 和 时 间 类 型 、 以 及 字符 串 类 型 。 请 密切 注意 它们 ， 因 为 
定义 数据 类 型 比 表 创建 过 程 中 的 任何 其 他 部 分 都 更 为 重要 。 

16.1.1 ”数字 数据 类 型 


MySQL 使 用 所 有 标准 的 ANSI SQL 数字 数据 类 型 ， 因 此 ， 如 果 我 们 
从 其 他 的 数据 库 系 统 转 到 MySQL， 这 些 定义 对 你 来 说 会 很 熟悉 。 如 下 
的 列表 给 出 了 常用 的 数字 数据 类 型 以 及 其 说 明 。 


提示 : 








数字 数据 类 型 的 列表 中 将 用 到 有 符 写 或 者 无 答 写 这 样 的 术语 。 如 果 你 还 记得 基本 的 代数 
知识 ， 就 会 知道 一 个 有 符号 整数 可 以 是 一 个 正 整数 或 者 负 整 数 ， 而 一 个 无 符号 的 整数 总 是 一 
个 非 负 的 整数 。 


e。 INT 一 一 一 个 常规 大 小 的 整数 ， 可 以 是 有 符号 的 或 者 无 符号 的 。 如 
果 是 有 符号 的 ， 人 允许 的 范围 从 -2147483648 到 2147483647。 如 果 是 
无 符号 的 ， 人 允许 的 范围 从 0 到 4294967295。 我 们 可 以 指定 最 大 11 位 
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TINYINT 一 一 一 个 小 的 整数 ， 可 以 是 有 符号 的 或 者 无 符号 的 。 如 
果 是 有 符号 的 ， 人 允许 的 范围 从 -128 到 127。 如 果 是 无 符号 的 ， 人 允许 
的 范围 从 0 到 255。 我 们 可 以 指定 最 大 4 位 的 宽度 。 

SMALLINT 一 一 一 个 小 的 整数 ， 可 以 是 有 符号 的 或 者 无 符 写 的 。 
如 果 是 有 符号 的 ， 人 允许 的 范围 从 -32768 到 32767。 如 了 果 是 无 符 扎 
的 ， 人 允许 的 范围 从 0 到 65535。 我 们 可 以 指定 最 大 5 位 的 宽度 。 
MEDIUMINT 一 一 一 个 中 等 大 小 的 整数 ， 可 以 是 有 符号 的 或 者 无 
人 符号 的 。 如 果 是 有 符号 的 ， 人 允许 的 范围 从 -8388608 到 8388607。 如 
果 是 无 符号 的 ， 人 允许 的 范围 从 0 到 16777215。 我 们 可 以 指定 最 大 9 位 
的 宽度 。 

BIGINT 一 一 一 个 较 大 的 整数 ， 可 以 是 有 符号 的 或 者 无 符号 的 。 如 
果 是 有 符号 的 ， 人 允许 的 范围 从 -9223372036854775808 到 
9223372036854775807。 如 末 是 无 符号 的 ， 人 允许 的 范围 从 0 到 
18446744073709551615。 我 们 可 以 指定 最 大 11 位 的 宽度 。 
FLOAT(M,D) 一 一 一 个 浮 点 数 ， 不 能 是 无 符号 的 。 我 们 可 以 定义 
显示 长 度 M) 和 小 数位 长 度 D) 。 但 这 不 是 必须 的 ， 默 认 值 为 
10,2， 其 中 2 是 小 数位 数 ， 而 10 是 总 位 数 〈 包 括 小 数位 ) 。 一 个 
FLOAT 的 小 数位 精度 可 以 达到 24 位 。 

DOUBLE(M,D) 一 一 一 个 双 精 上 度 浮 点 数 ， 不 能 是 无 从 写 的 。 我 们 可 
以 定义 显示 长 度 M) 和 小 数位 长 度 D) 。 但 这 不 是 必须 的 ， 默 
认 值 为 16,4， 其 中 4 是 小 数位 数 。 一 个 DOUBLE 的 小 数位 精度 可 以 
达到 53 位 。REAL 是 DOUBLE 的 同义词 。 

DECIMAL(M,D) 一 一 一 个 未 打包 的 双 精 度 浮 点 数 ， 不 能 是 无 符号 
的 。 在 未 打包 的 小 数 中 ， 每 个 小 数 对 应 一 个 字 贡 。 定 义 显示 长 度 


(M) 和 小 数位 长 度 D) 是 必须 的 。NUMERIC 是 DECIMAL 的 同 
义 词 。 


在 所 有 的 MySQL 数 字数 据 类 型 中 ， 最 经 党 使 用 INT。 如 果 我 们 定义 
自己 的 字段 比 实 际 所 需 的 小 ， 可 能 会 遇 到 问题 。 人 例如， 如果 我 们 把 一 个 
ID 字段 定义 为 无 符号 的 TINYINT， 如 条 ID 是 一 个 主键 〈 并 且 是 必须 的 字 
BO) ， 我 们 将 不 能 成 功 地 插入 第 256 条 记录 。 


16.1.2 ”日 期 和 时 间 类 型 


MySQL 有 有 几 种 数据 类 型 可 以 用 来 存储 日 期 和 时 间 ， 这 些 数 据 类 型 
在 输入 方面 很 灵活 。 换 名 话说， 我 们 可 以 输入 那些 并 不 是 真正 的 日 子 的 
日 期 ， 例 如 2 月 30 写 ， 而 2 月 只 有 28 或 29 写 ， 没 有 30 写 。 男 外 ， 我 们 可 以 
存储 带 有 遗失 的 信息 的 日 期 。 例 如 ， 如 果 我 们 知道 菜 人 出 生 于 1980 年 11 
月 的 某 天 ， 可 以 使 用 1980-11-00， 而 00 表 示 出 生 的 日 期 ， 如 果 我 们 知道 
日 期 的 话 。 


MySQL 的 日 期 和 时 间 类 型 的 灵活 性 也 意味 看 ， 日 期 检查 的 职 贡 沙 
到 了 应 用 程序 开发 者 的 肩 上 。MySQL 只 检查 两 个 元 素 的 有 效 性 : 月 份 
是 人 盏 在 0 到 12 之 则 ， 以 及 日 期 在 0 到 31 之 则 。MySQL 个 会 目 动 验证 2 月 的 
30 写 是 合 是 有 有效 的 日 期 。 因 此 ， 应 用 程序 内 所 要 进行 的 任何 日 期 验证 ， 
邵 应 该 在 PHP 代 人 码 中 进行 ， 而 且 在 试图 用 假 的 日 期 间 数 据 库 表 添加 一 条 
记录 之 前 束 进 行 验证 。 


MySQL 的 日 期 和 时 间 数 据 类 型 如 下 。 


e DATE 一 一 YYYY-MM-DD 格 式 的 一 个 日 期 ， 在 1000-01-01 到 9999- 
12-31 之 间 。 例 如 ，1973 年 12 月 30 日 ， 将 存储 为 1973-12-30。 


FA 


DATETIME ——YYYY-MM-DD HH:MM:SS 格 式 的 一 个 日 期 和 时 
间 组 合 ， 在 1000-01-01 00:00:00 到 9999-12-31 23:59:59 之 间 。 例 如 ， 
1973 年 12 月 30 日 下 午 3:30 将 存储 为 1973-12-30 15:30:00。 
TIMESTAMP 一 一 1970 年 1 月 1 日 午夜 和 2037 年 某 个 时 间 之 间 的 一 
个 时 间 惟 。 我 们 可 以 为 TIMESTAMP 定 义 多 个 长 度 ， 这 直接 和 其 中 
存储 的 内 容 相 关 。TIMESTAMP 缺 省 的 长 度 是 14， 其 中 存储 了 
YYYYMMDDHHMMSS。 这 看 上 去 和 前 面 的 DATETIME 格 式 相 
似 ， 只 是 在 数字 之 间 没 有 连 字 符号 。1973 年 12 月 30 日 下 午 3:30， 将 
存储 为 19731230153000。TIMESTAMP 的 其 他 定义 是 12 
(YYMMDDHHMMSS)、8 (YYYYMMDD)J 和 6 (YYMMDD)。 

TIME 一 以 HH:MM:SS 格 式 存储 时 间 。 

YEAR(M) 一 一 以 2 位 或 4 位 的 格式 存储 年 份 。 如 末 长 度 指 定 为 2〈 例 
如 YEAR(2)) ，YEAR 可 以 是 1970 到 2069 〈70 到 69) 。 如 果 长 度 指 
定 为 4，YEAR 可 以 是 1901 到 2155。 默 认 长 度 是 4。 


DATETIME 或 DATE 比 其 他 的 日 期 和 时 间 相 关 的 数据 次 型 更 为 第 


16.1.3 ”字符 串 类 型 


尽管 数字 和 日期 类 型 很 有 趣 ， 但 我 们 所 存储 的 大 多 数 数据 将 是 字符 


串 格式 的 。 这 里 列 出 了 MySQL 中 常用 的 字符 串 数据 类 型 。 


e CHAR(M) 一 一 一 个 定 长 的 字符 串 ， 长 度 在 1 到 255 个 学 和 从 之 则 ， 例 


如 CHAR(5)。 存 储 的 时 候 ， 石 按 使 用 空 日 填充 到 指定 的 长 度 ， 定 义 
时 长 度 不是 必须 的 ， 但 献 认为 1。 


。VARCHAR(M) 一 一 一 个 变 长 的 字符 串 ， 长 度 在 1 到 255 个 字符 之 


半 ， 例 如 VARCHAR(25)。 在 创建 一 个 VARCHAR 字 段 的 时 候 ， 必 
须 定 义 一 个 长 度 。 

BLOB 或 TEXT 最 大 长 度 为 65535 个 字符 的 一 个 字段 。BLOB 表 
示 “Binary Large Objects” 《二进制 大 对 象 ) ， 并 且 用 来 存储 大 容量 
的 二 进 制 数据 ， 例 如 岁 像 或 者 其 他 次 型 的 文件 。 定 义 为 TEXT 的 字 
段 也 存储 大 量 的 数据 。 二 者 之 间 的 不 同 在 于 ， 对 于 存储 的 数据 的 排 
序 和 比较 ， 在 BLOB 上 十 区 分 大 小 写 的 ， 而 在 TEXT 字段 上 是 不 区 
分 大 小 写 的 。 我 们 不 对 BLOB 或 TEXT 指定 长 度 。 

TINYBLOB 或 TINYTEXT 一 一 最 大 长 虎 为 255 个 字符 的 一 个 BLOB 
或 TEXT。 我 们 不 对 TINYBLOB 或 TINYTEXT 指 定 一 个 长 度 。 
MEDIUMBLOB 或 MEDIUMTEXT 最 大 长 度 为 16777215 个 字 
符 的 一 个 BLOB 或 TEXT。 我 们 不 对 MEDIUMBLOB 或 
MEDIUMTEXT 指 定 一 个 长 度 。 

LONGBLOB 或 LONGTEXT 一 一 最 大 长 度 为 4294967295 个 字符 的 
一 个 BLOB 或 TEXT。 我 们 不 对 LONGBLOB 或 LONGTEXT 指 定 一 个 
长 度 。 

ENUM 一 一 一 个 枚 淮 类 型 ， 即 指定 项 目的 一 个 列表 。 当 定义 一 个 
ENUM 的 时 候 ， 我 们 创建 了 一 个 项 目的 列表 ， 值 必须 从 这 个 列表 中 
选 定 或 者 为 NULL。 例 如 ， 如 果 希 望 自己 的 字段 包 

含 “A” 或 “B” 或 “C”， 我 们 可 以 定义 ENUM 为 ENUM (‘A’, ‘B’, ‘C’), 
则 只 有 这 些 值 或 者 NULL 可 以 填 入 到 这 个 字段 。ENUM 可 以 有 65535 
个 不 同 的 但 。ENUM 使 用 一 个 索引 来 存储 项 目 。 











SET 类 型 和 ENUM 类 型 相似 ， 因 为 它 也 定义 了 一 个 列表 。 然 而 ，SET 类 型 存储 为 一 个 完整 


的 值 ， 而 不 是 像 ENUM 那 样 ， 存 储 为 一 个 值 的 一 个 索引 。 





VARCHAR 和 TEXT 字段 比 其 他 的 字段 类 型 更 为 向 用， 而 ENUM 也 
很 有 用 。 


16.2” 表 的 创建 语法 
BEATS fi 2a RIL BR. 


© RINKS. 
。 字段 的 名 字 ， 
。 每 个 字段 的 定义 ， 


创建 表 的 一 般 语 法 如 下 。 


CREATE TABLE table name (column name column type}; 


表 名 取决 于 你 ， 但 是 应 该 是 能 够 反应 表 的 用 途 的 名 字 。 例 如 ， 如 末 
你 有 一 个 表 用 来 存储 一 个 林 仙 店 的 存货 ， 不 应 该 把 这 表 命 名 为 s 而 应 访 
将 其 命名 为 类 似 grocery_inventory 的 名 字 。 类 似 的 ， 我 们 所 选择 的 字段 
名 也 应 该 尽 可 能 地 精 烧 ， 并 且 和 它们 所 起 的 作用 以 及 它们 所 存储 的 数据 
相关 。 例 如 ， 我 们 可 能 把 体 存 商品 名 字 的 一 个 字段 命名 为 item_name， 
而 个 十 n。 


下 面 的 例子 创建 了 一 个 通用 的 grocery_inventory 表 ， 它 保存 了 用 于 
商品 ID、 商 品名 称 、 商 品 介绍 、 定 价 和 数量 的 字段 。 这 些 字 段 的 数据 区 
型 各 不 相同 ，ID 和 数量 字段 保存 整数 ， 商 品名 称 字 段 最 多 保存 50 个 字 
和 侍 ， 商 品 介绍 字段 最 多 保存 65535 个 文本 字符 ， 而 定价 字段 保存 一 个 浮 
上 尽数 ， 不 例如 下 。 


CREATE TABLE grocery_inventory ( 
id INT NOT NULL PRIMARY KEY AUTO_INCREMENT, 
item_name VARCHAR (50) NOT NULL, 
item desc TEXT, 
item_price FLOAT NOT NULL, 
curr_qty INT NOT NULL 


); 
提示 : 
id 字 上 段 定 义 为 一 个 主键 。 我 们 将 在 后 面 的 草 市 中 ， 即 在 把 具体 的 表 作 为 示例 应 用 程序 的 一 


部 分 创建 的 环境 中 ， 学 习 有 关键 的 内 容 。 通 过 使 用 auto_increment 作 为 字段 属性 ， 我 们 告诉 
MySQL 继 续 前 进 并 且 把 下 一 个 可 用 的 编号 作为 ID 字段 的 值 。 





MySQL 服 务 器 使 用 Query OK 啊 应 一 条 成 功 执行 的 命令 ， 而 不 管 这 
条 命令 的 类 型 是 什么 。 否 则 ， 将 会 显示 一 条 错误 消息 ， 告 诉 你 查询 出 
着。 根据 你 所 使 用 的 界面 的 不 同 ， 你 可 能 会 也 可 能 不 会 看 到 这 条 特定 的 
啊 应 。 然 而 ， 不 管 使 用 什么 界面 ， 它 应 该 都 会 给 出 和 查询 状态 相关 的 提 
ZN o 


16.3 ”使 用 INSERTI 命 仿 


在 创建 了 一 些 表 后 ， 可 以 使 用 SQL 命 令 INSERT 来 回 这 些 表 江 加 新 
的 记录 。INSERT 的 基本 语法 如 下 。 


INSERT INTO table name (column list) VALUES (column values); 


在 括 写 中 的 值 列表 中 ， 我 们 必须 使 用 引号 括 起 字符 串 。SQL 标 准 是 
持 引 写 ， 但 MySQL 人 允许 使 用 时 引号 或 者 双 引 写 。 如 果 引 写 在 字符 串 本 
AZ, Alls SOT AT ALE SI] Ss WY SAS ET eX 


提示 : 


整数 不 需要 使 用 引号 括 起 来 。 


下 面 是 一 个 需要 转 义 的 字符 串 的 例子 。 
O'Connor said "Boo" 
如 条 我 们 把 字符 串 放 入 到 双 引 号 中 ，INSERT 语 句 将 会 如 下 所 示 。 
INSERT INTO table name (column_name) VALUES {"0'Connor said \"Boo\""); 
如 果 我 们 把 字符 串 放 入 到 一 个 单 引号 中 ，INSERT 语 句 将 会 如 下 所 
示 。 
INSERT INTO table name (column_name) VALUES ('0\'Connor said "Boo"'); 


进一步 学 习 INSERT 语 名 


除了 表 的 名 字 ，INSERT 语 句 中 还 有 两 个 重要 的 部 分 ， 即 列 列表 和 


值 列 表 。 只 有 值 列表 是 必须 的 ， 但 是 ， 如 果 省 略 了 列 列表 ， 必 须 严格 按 
照 对 应 的 顺序 在 值 列 表 中 为 这 些 列 指定 值 。 


以 grocery_inventory 表 为 例 ， 我 们 有 5 个 字段 : id、item_name、 
item_desc、item_price 和 curr_ qty。 要 插入 一 条 完整 的 记录 ， 可 以 使 用 如 
下 这 些 语句 中 的 任何 一 条 。 


。 市 有 所 有 列 名 的 一 条 语句 ， 示 例如 下 。 


INSERT INTO grocery_inventory 
{id, item name, item desc, item price, curr_qty) 
VALUES ( 1 ， ‘Apples’, ‘Beautiful, ripe apples.', ‘0.25', 1000}; 


。 使 用 所 有 列 ， 但 不 显 式 指定 这 些 列 的 一 条 语句 ， 示 例如 下 。 


INSERT INTO grocery inventory VALUES ( 2 ， Bunches of Grapes’, 
‘$eedless grapes.', ‘2.99', 500}; 


iN PIXAR), BERRES. WA Aare 
Æ “Query OK” R- 


现在 ， 介 绍 使 用 INSERT 的 一 些 更 为 有 趣 的 方法 。 由 于 ID 是 一 个 自 
动 增加 的 整数 ， 我 们 不 必 将 其 放 入 到 值 列 表 中 。 然 而 ， 如 果 有 一 个 值 我 
们 有 意 不 想 列 出 ， 例 如 id， 那 必须 把 后 续 使 用 到 的 其 他 列 都 列 出 来 。 例 
如 ， 如 下 的 语句 没有 给 出 列 列 表 ， 并 且 也 没有 给 id 一 个 值 。 


INSERT INTO grocery inventory VALUES 
(‘Bottled Water (6-pack})', ‘580ml Spring water.', 2.29, 250); 


上 述 语句 或 产生 如 下 的 一 个 错误 。 
ERROR 1136: Column count doesn't match value count at row 1 


由 于 没有 列 出 任何 列 ，MySQL 期 每 它们 部 出 现在 值 列表 中 ， 从 而 


导致 亲 面 一 条 语句 产生 错误 。 如 果 目 标 是 让 MySQL 通 过 上 自动 增加 id 字段 
来 为 你 完成 工作 ， 我 们 可 以 使 用 下 面 这 些 语句 中 的 一 条 。 


。 市 有 除了 id 以 外 的 所 有 列 名 的 一 条 语句 。 


INSERT INTO grocery inventory (item name, item desc, item price, curr dty) 
VALUES (‘Bottled Water (6-pack)', ‘50@ml spring water.', ‘2.29', 250); 


NULL 值 以便 MySQL 可 以 上 自动 扰 入 一 个 值 。 


INSERT INTO grocery_inventory VALUES ( NULL ， ‘Bottled Water (12-pack}', 
‘500ml spring water. , 4.49, 500}; 


这 两 条 语句 都 答 试 一 下 ， 这 样 grocery_inventory 表 一 共 束 有 了 4 条 记 
录 了。 选择 哪 条 语句 对 于 MySQL 来 说 没有 什么 区 别 ， 这 是 你 个 人 的 偏 
好 而 已 ， 但 是 应 在 你 的 应 用 程序 开 有 友 中 保持 一 致 。 一 致 的 结构 将 会 使 你 
随后 的 调试 工作 更 容易 ， 因 为 你 将 知道 预期 得 到 什么 结 


16.4 使 用 SELECT 命令 


SELECT 是 用 来 从 表 中 获取 数据 的 命令 。 这 条 命令 的 语法 可 以 非常 
简单 也 可 能 非常 复 淋 ， 取 决 于 你 所 要 选择 的 字段 ， 你 是 否 从 多 个 表 中 选 
取 ， 以 及 你 计划 施加 什么 条 件 。 随 着 对 数据 库 程序 设计 越 来 越 熟 悉 ， 你 
将 会 学 习 扩 展 SELECT 语 句 ， 最 终 使 得 你 的 数据 库 做 尽 可 能 多 的 工作 而 
不 会 让 编程 语言 负担 过 重 。 


最 基本 的 SELECT 语法 如 下 所 示 。 


SELECT expressions and columns FROM table name 
[WHERE some condition is true] 

[ORDER BY some column [ASC ! DESC]] 

[LIMIT offset, rows] 


看 看 其 中 第 一 行 ， 内 容 如 下 。 
SELECT expressions and columns FROM table name 

一 个 方便 的 表达 式 束 是 * 符 写 ， 它 表示 一 切 内 容 。 因 此 ， 要 但 询 
grocery_inventory 表 中 的 一 切 内 容 (所 有 行 、 所 有 列 ) ，SQL 语 句 如 
下 。 


SELECT * FROM grocery_inventory; 


根据 在 grocery_inventory 中 找到 的 数据 的 多 少 ， 我 们 的 结果 可 能 
不 相同 ， 但 是 ， 结 果 会 如 下 所 示 。 


Pears SLMS SMe, gees pees Se PUR SS Lok Pes ees se FALARO Dae SOILD LAG ES, NETS + 
| id | item_name | item_desc | item_price | curr_qty | 
ee Sm ee ee. BR ME Prega antics Gates SRA a E T on + 
| 1 | Apples | Beautiful, ripe apples. | 0.25 | 1800 | 
| 2 | Bunches of Grapes | Seedless grapes. | 2.99 | 500 | 
| 3 | Bottled Water (6-pack}) | 5@@ml spring water. | 2.29 | 250 | 
| 4 | Bottled Water (12-pack} | 500ml spring water. | 4.49 | 500 | 
+----+------------------------- +------------------------- +------------ +---------- 十 


4 rows in set (0.00 sec} 


这 是 MySQL 命 令 行 界 面 的 输出 ， 你 可 以 看 到 ，MySQL 创 建 了 一 个 
可 爱 的 、 格 式 化 的 表 ， 使 用 结果 集 的 第 一 行 作为 表 的 列 名 。 如 果 你 使 用 
不 同 的 MySQL 界 面 ， 结 果 看 上 去 会 有 所 不 同 〈 注 意 观 察 期 望 的 数据 ， 
而 不 是 界面 的 不 同 ) 。 


如 末 我 们 只 想 选 择 特定 的 列 ， 使 用 列 名 来 奉 代 *， 多 个 列 名 之 间 用 
和 逗号 隔 开 。 如 下 的 语句 只 从 grocery_inventory 表 选择 id、item_name 和 
curr_dty 字 段 。 


SELECT id, item_name, curr_qty FROM grocery inventory; 


结 末 显示 如 下 所 示 。 
Er SN a Sees RE see ME LES oe + 
| 1d | item_name | curr_qty | 
于 was ee + 
| 1 | Apples | 1008 | 
| 2 | Bunches of Grapes | 500 | 
| 3 | Bottled Water (6-pack) | 250 | 
| 4 | Bottled Water (12-pack) | 500 | 
+----+------------------------- +---------- 十 


4 rows in set (0.90 sec) 


16.4.1 排序 SELECT 结果 


SELECT 得 询 的 结束 默认 按照 它们 插入 到 和 中 的 顺序 来 排序 ， 并 且 
不 会 依赖 于 一 个 有 意义 的 排序 系统 。 如 采 我 们 想 要 按照 一 种 特定 方式 对 
结 朱 排序 ， 人 例如， 按照 日 期 、ID、 名 字 等 等 来 排序 ， 需 要 使 用 ORDER 


BY 子 句 来 明确 我 们 的 需求 。 在 下 面 的 语句 中 ， 结 果 按 照 item_name 的 字 
母 顺 序 来 排序 。 


SELECT id, item name, curr qty FROM grocery inventory 
ORDER BY item name; 


执行 成 功 的 结 末 如 下 所 示 。 


A DON STE tere ms DAT + 
| 1d | item_name | curr_qty | 
ieee Sarees omens See aun cect ea 十 
| 1 | Apples | 1008 | 
| 4 | Bottled Water (12-pack) | 500 | 
| 3 | Bottled Water (6-pack) | 250 | 
| 2 | Bunches of Grapes | 500 | 
+----+------------------------- +---------- 十 


4 rows in set (0.93 sec) 


你 知道 吗 ? 


当 从 表 中 和 奉 询 了 结果 而 没有 指定 排列 顺序 的 时 候 ， 结 采 可 能 按照 键 值 排 序 ， 也 可 能 不 这 
样 排序 。 由 于 MySQL 重 新 使 用 了 前 面 删除 的 行 押 占用 的 空间 ， 融 会 有 友 生 这 种 情况 。 换 句 话 
说 ， 如 来 我 们 添加 了 TD 值 从 1 到 5 的 记录 ， 删 际 了 了 D 为 4 的 记录 ， 然 后 义 添加 了 为 外 一 条 记录 
GD 为 6) ， 记 录 可 能 按照 下 面 的 顺序 出 现在 表 中 。 


1, 2, 3, 6, 5. 


ORDER BY 默认 的 排序 是 升序 (ASC) ， 字 符 串 顺序 是 从 A 到 Z， 
整数 顺序 是 从 0 开始 ， 日 期 顺序 是 从 最 早 的 日 期 到 最 近 的 日 期 。 也 可 以 
指定 一 个 降序 ， 使 用 DESC， 示 例如 下 。 


SELECT id, item_name, curr_qty FROM grocery_inventory 
ORDER BY item_name DESC; 


执行 的 结果 如 下 。 


Prt peas SS ees Sees. Soe See FASET eS Se + 
| 1d | item_name | curr_qty | 
下 is nos aie 十 
| 2 | Bunches of Grapes | 500 | 
| 3 | Bottled Water (6-pack) | 250 | 
| 4 | Bottled Water (12-pack) | 500 | 
| 1 | Apples | 1000 | 
+----+------------------------- +---------- 十 


4 rows in set (0.90 sec) 


我 们 并 没有 受 限 于 只 对 一 个 字段 排序 ， 可 以 指定 尽 可 能 多 的 字段 ， 
字段 之 间 用 喜 写 阳 开 。 排 序 优 先 级 按照 列 出 的 季 段 的 顺序 。 


16.4.2 ”上 限制 结果 


可 以 使 用 LIMIT 子 句 来 从 SELECT 查 询 结 果 中 返回 一 定数 目的 记 
录 。 使 用 LIMIT 子 句 的 时 候 可 以 有 两 个 参数 : 偏 移 量 和 行 数 。 偏 移 量 是 
起 始 位 置 ， 而 行 数 应 该 是 目 说 明 的 。 


假设 grocery_inventory 表 中 有 多 于 两 或 三 条 记录 ， 并 且 我 们 希望 选 
择 按 照 curr_qty 排 序 的 前 两 条 记录 的 ID、name 和 quantity。 换 句 话 说 ， 我 
们 斋 望 选取 存货 最 少 的 两 项 。 如 下 的 单 参数 限制 将 会 从 0 位 置 开 始 ， 并 
且 到 达 第 二 条 记录 。 


SELECT id, item_name, curr_qty FROM grocery_inventory 
ORDER BY curr_qty LIMIT 2; 


执行 的 结果 如 下 。 
Se aos Bees Sees + 
| id | item_name | curr_qty | 
Foto ee Fe eee + 
| 3 | Bottled Water (6-pack) | 250 | 
| 2 | Bunches of Grapes | 500 | 
Rc e een ne Eana i ee + 


2 rows in set (0.00 sec} 


LIMIT 了 于 人 句 在 实际 的 应 用 中 很 有 有 用。 例如， 我 们 可 以 在 一 组 


SELECT 语句 中 使 用 LIMIT 来 分 步骤 过 历 络 果 。 首 先是 头 两 项 ， 然 后 是 
接 下 来 的 两 项 ， 最 后 又 是 接 下 来 的 两 项 ， 示 例如 下 。 


1. SELECT * FROM grocery_inventory ORDER BY curr_qty LIMIT 
0, 2; 


2. SELECT * FROM grocery_inventory ORDER BY curr_gty LIMIT 
2,2; 


3. SELECT * FROM grocery inventory ORDER BY curr_gty LIMIT 
4, 2; 


如 果 在 查询 中 指定 了 行 的 一 个 偏 移 量 和 数目 ， 但 没有 找到 结果 ， 我 
们 也 不 会 看 到 错误 ， 而 只 是 看 到 一 个 空 的 结果 集 。 例 如 ， 如 果 
grocery_inventory 表 只 包含 了 6 条 记录 ， 一 条 市 有 一 个 俩 移 量 为 6 的 LIMIT 
的 查询 将 不 会 产生 结果 。 


在 基于 Web 的 应 用 程序 中 ， 当 数据 的 列表 通过 一 个 “前 10 条 ”和 “后 
10 条 ”之 类 的 链接 来 显示 的 时 候 ， 这 很 可 能 就 需要 使 用 LIMIT 子 句 。 


16.5 ”在 但 询 中 使 用 WHERE 


我 们 已 经 学 习 了 很 多 方法 来 从 表 中 获取 特定 的 列 ， 但 还 没有 获取 指 
定 的 行 ， 而 这 正 是 WHERE 子 句 的 用 武之 地 。 从 SELECT 语法 的 例子 
中 ， 我 们 看 到 了 WHERE 用 来 指定 一 个 特定 的 条 件 。 


SELECT expressions and columns FROM table name 
[WHERE some condition is true] 


下 面 的 一 个 例子 是 ， 获 取 那 些 数量 为 500 的 商品 的 记录 。 


SELECT * FROM grocery inventory WHERE curr qty = 500; 


结果 如 下 所 示 。 
Pires Se Ev tau Sea Se toni wees 人 JUUTE uona th + 
| id | item_name | item_desc | item_price | curr_qty | 
foes fee ee ee ee {eee Oe Sere face ware i Tousa RS m + 
| 2 | Bunches of Grapes | Seedless grapes. | 2.99 | 500 | 
| 4 | Bottled Water {12-pack) | 50@ml spring water. | 4.49 | 500 | 
和 aoni aie o nu en en ee + 


2 rows in set (0.00 sec) 


EWA RANEY, BOAR BRANT EA — PERU ASWHERE FJ HN — 
may, JPN SS HEALS SIDS A BRR, SPR Ee 
用 同样 的 规则 ， 这 些 规则 我 们 已 在 十 面 有 关 INSERT 的 小 让 中 学 习 过 。 


16.5.1 在 WHERE 子 句 中 使 用 操作 符 


我 们 已 经 在 WHERE 子 人 句 中 使 用 了 相等 操作 符 “=) 来 确定 一 个 条 件 
的 真 ， 也 束 是 说 ， 一 个 事物 是 任 等 于 为 外 一 个 事物 。 我 们 可 以 使 用 多 种 
操作 从 ， 其 中 ， 比 较 操作 行 和 效 辑 操作 从 是 最 为 昭 用 的 类 型 。 表 16-1 列 
出 了 比较 操作 符 及 它们 的 含义 。 


416-1 基本 比较 操作 符 及 其 含义 


小 于 或 等 于 





还 有 一 个 叫做 BETWEEN 的 方便 的 操作 从 ， 它 在 比较 整数 或 数据 的 
时 候 很 有 用 ， 因 为 它 搜索 位 于 一 个 最 小 值 和 最 大 值 之 间 的 结果 ， 示 例如 
Ts 


SELECT * FROM grocery_inventory WHERE item price BETWEEN 1.50 AND 3.00; 


结果 如 下 所 示 。 
ee A ne a DEE J Se. J TO DS a Pee iE paves SSK E 十 
| id | item name | item desc | item price | curr_qty | 
Fs A Lr Me ren en + 
| 2 | Bunches of Grapes | Seedless grapes. | 2.99 | 500 | 
| 3 | Bottled Water (6-pack) | 50@ml spring water. | 2.29 | 250 | 
Bi ee Se DT a Fe Se oe » ae, Sl Oa feng + 


2 rows in set (0.00 sec) 


其 他 的 操作 符 包 括 逻 辑 操作 符 ， 它 们 使 得 我 们 可 以 在 WHERE 子 名 
中 使 用 多 个 比较 。 基 本 好 辑 操 作 符 是 AND 和 OR。 当 使 用 AND 的 时 候 ， 
子 句 中 的 所 有 比较 结果 必须 是 真 才 能 得 到 结果 ， 而 使 用 OR 则 允许 人 至少 
有 一 个 比较 结果 为 真 。 另 外 ， 我 们 可 以 使 用 IN 操作 人 符 来 指定 想 要 匹配 的 
商品 的 一 个 列表 。 


16.5.2 ”使 用 LIKE 比 较 字 符 串 


前 面 已 经 介绍 了 在 一 个 WHERE 子 句 中 通过 使 用 = 或 != 来 匹配 字符 
昌 ， 但 是 ， 还 有 万 外 一 种 有 用 的 换 作 符 可 以 用 来 在 WHERE 子 句 中 比较 
字符 种 ， 这 融 是 LIKE 操 作 符 。 这 个 操作 符 在 模 却 匹配 中 可 以 使 用 如 下 
两 个 字符 作为 通配符 。 





e % 多 配 多 个 字符 。 
e ——_[LAt—-7 FFF. 


例如 ， 如 果 我 们 想 要 在 grocery_inventory 表 中 找到 名 字 的 第 一 个 字 
母 为 A 的 商品 ， 可 以 使 用 如 下 语句 。 


SELECT * FROM grocery inventory WHERE item_name LIKE ‘A*%'; 


结果 如 下 所 示 。 
Tere maths. ESE 2 eRe Es a ee es Se rie, Er es Oe + 
| id | item name | item desc | item price | curr qty | 
+----+----------- +------------------------- +------------ +---------- 十 
| 1 | Apples | Beautiful, ripe apples. | 0.25 | 1000 | 
ee fee ee ee eee ea eee eee fee ee eee ee + 


1 row in set {0.00 sec} 
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除非 在 一 个 二 进 制 字符 串 上 执行 一 个 LIKE 比 较 ， 人 否则 这 个 比较 总 是 不 区 分 大 小 写 的 。 我 
们 可 以 使 用 BINARY 关 键 字 来 强制 执行 一 个 区 分 大 小 写 的 比较 。 





16.6 MŚ TK Rw 


我 们 并 没有 受到 限制 只 能 同时 查询 一 个 表 。 如 果 是 那样 的 话 ， 应 用 
程序 编程 将 会 古 一 个 漫长 而 枯燥 的 任务 。 妆 我 们 在 一 条 SELECT 语 句 中 
对 多 个 表 合 询 的 时 候 ， 确 实 会 把 这 些 表 连 接 到 一 起 。 


假设 我 们 有 两 个 表 : fruit 表 和 color 表 。 我 们 可 以 使 用 两 条 单独 的 
SELECT 语 句 ， 分 别 从 这 两 个 表 中 的 每 一 个 来 查询 行 ， 示 例如 下 。 


SELECT * FROM fruit; 


IX A Va er" OP 


1 

1 

1 

1 

1 
于 


fruitname | 
eee eee eee 一 


grape 


路 
i 
+ 一 一 一 一 + 一 十 


| 
orange | 

| 

| 
4 rows in set (0.00 sec) 


SELECT * FROM color; 


二 条 但 询 会 产生 如 下 结果 。 


ES SD wee + 
| colorname | 
Fees dn se, a + 
| | 
| orange | 
| purple | 
| | 


4 rows in set (0.00 sec) 


当 我 们 想 要 一 次 从 两 个 表 碍 询 数据 的 时 候 ，SELECT 语 句 在 语法 上 


略 有 不 同 。 首 先 ， 必 须 确保 在 查询 中 所 使 用 的 所 有 的 表 都 出 现在 
SELECT 语句 的 FROM 子 句 中 。 以 fruit 和 color 为 例 ， 如 果 只 是 想 从 两 个 
表 中 获取 所 有 的 列 和 行 ， 你 可 能 会 考虑 使 用 如 下 的 SELECT 语句 。 


SELECT * FROM fruit, color; 


使 用 这 条 碍 询 ， 我 们 会 得 到 如 下 结 朱 。 


E ee ee E + 
| id | fruitname | id | colorname | 
pee eee eee Se Se + 
| 1 | apple | 1 | red | 
| 2 | orange | 1 | red | 
| 3 | grape | 1 | red | 
| 4 | banana | 1 | red | 
| 1 | apple | 2 | orange | 
| 2 | orange | 2 | orange | 
| 3 | grape | 2 | orange | 
| 4 | banana | 2 | orange | 
| 1 | apple | 3 | purple | 
| 2 | orange | 3 | purple | 
| 3 | grape | 3 | purple | 
| 4 | banana | 3 | purple | 
| 1 | apple | 4 | yellow | 
| 2 | orange | 4 | yellow | 
| 3 | grape | 4 | yellow | 
| 4 | banana | 4 | yellow | 
een ee ee eee ages Hi heel eta se He ab te + 


16 rows in set (@.@@ sec) 


16 行 重复 的 信息 可 能 不 是 我 们 所 要 奉 找 的 。 这 个 得 询 所 做 的 是 把 
color 表 中 的 每 一 行 逐 个 地 连接 到 fruit 表 中 的 每 一 行 。 由 于 fruit 表 中 有 4 条 


当 我 们 想 从 多 个 表 选 择 的 时 候 ， 必 须 构 建 正 确 的 WHERE 子 句 来 确 
保 确 实 能 够 得 到 想 要 的 记录 。 在 fruit 和 color 表 的 例子 中 ， 我 们 真正 想 要 
的 是 看 到 ID 相同 的 来 自 两 个 表 中 的 fruitname 和 colorname 记 录 。 这 给 我 们 
市 来 了 下 一 个 略 有 不 同 的 得 询 一 一 当 字 段 在 爽 个 表 中 的 名 字 都 相同 的 时 


IR WAR Ae aN BYE EBNF FB 


Rta, BONA i ees TRAMAI ERAMI SO, UR PAN 


tablename.fieldname 


这 样 ， 从 两 个 表 中 和 查询 ID 相 等 的 fruitname 和 colorname 的 查询 就 是 如 
ies 


SELECT fruitname, colorname FROM fruit, color WHERE fruit.id = color.id; 


IX Ba are EE ZR, AN BIT FE 


To Been piste sees See + 
| fruitname | colorname | 
a Fema daune pis + 
| apple | red | 
| orange | orange | 
| grape | purple | 
| banana | yellow | 
+----------- +----------- 十 


4 rows in set (0.00 sec) 


然而 ， 如 条 我 们 试图 碍 找 在 两 个 表 中 都 出 现 的 且 名 字 相 同 的 一 个 
列 ， 束 会 得 到 一 个 二 义 性 的 错误 。 


SELECT id, fruitname, colorname FROM fruit, color 
WHERE fruit.id = color.id; 


IKE A ee OB FR 


ERROR 1052: Column: ‘id’ in field list is ambiguous 


如 果 是 想 要 从 fruit 表 中 获取 ID， 使 用 如 下 语句 。 


SELECT fruit.id, fruitname, colorname FROM fruit, 
color WHERE fruit.id = color.id; 


IK A Val ar OP RW 


spans te se AEN sf aL AA Sree + 
| id | fruitname | colorname | 
ES et 二 十 
| 1 | apple | rea | 
| 2 | orange | orange | 
| 3 | grape | purple | 
| 4 | banana | yellow | 
PURGERI ee 下 十 


4 rows in set {@.0@ sec) 


IX ETE PN PACE ETE E EEA H SELECT 2 W PEH H 
一 个 基本 的 例子 。JOIN 关 键 字 实际 上 是 SQL 的 一 部 分 ， 它 使 得 我 们 能 够 
构建 更 为 复杂 的 查询 。 


16.6.1 ”使 用 JOIN 


在 MySQL 中 ， 有 几 种 类 型 的 JOIN 可 供 使 用 ， 所 有 的 这 些 都 涉及 到 
表 组 合 到 一 起 的 顺序 以 及 结果 显示 的 顺序 。 与 fruit 表 和 color 表 一 起 使 用 
的 JOIN 叫做 INNER JOIN， 尽 管 不 必 显 式 地 这 样 号 。 要 使 用 正确 的 
INNER JOIN 语法 来 重新 编导 上 述 SQL 语 句 ， 示 例如 下 。 


SELECT fruitname, colorname FROM fruit 
INNER JOIN color ON fruit.id = color.id; 


结果 如 下 所 示 。 
DS Pose ponui Po + 
| fruitname | colorname | 
pometa Senideei + 
| apple | red | 
| orange | orange | 
| grape | purple | 
| banana | yellow | 
+----------- +----------- 十 


4 rows in set (0.00 sec) 


ONS AAR T BTA ALS ESWHERE TE, FER PIP, € 


告诉 MySQL 把 表 中 ID 相 匹 配 的 行 连接 起 来 。 当 使 用 ON 子 句 连 接 表 的 时 
候 ， 可 以 使 用 那些 能 够 在 WHERE 子 句 中 使 用 的 任何 和 条件， 包括 所 有 各 
种 逻辑 操作 符 和 算术 操作 符 。 


另 一 种 常见 的 JOIN 类 型 是 LEFT JOIN。 使 用 LEFT JOIN 把 两 个 表 连 
接 到 一 起 的 时 候 ， 第 一 个 表 中 的 所 有 的 行 都 将 返回 ， 不 官 它 在 第 二 个 表 
中 是 个 有 匹配 。 假 设 在 一 个 地 址 短 中 有 两 个 表 ， 一 个 叫做 
master_ name， 包 含 基 本 的 记录 ， 一 个 叫做 email， 包 括 Email 记 录 。email 
表 中 的 任何 记录 都 将 绑 定 到 master_name 表 中 的 一 条 记录 的 一 个 特定 
ID。 首 先 看 看 这 两 个 表 的 内 容 。 


pees, Hees Se cf Meee 2 2 + 
| name_id | firstname | lastname | 
下 Foer ses Sake T as eraai ia + 
| 1 | John | Smith | 
| 2 | Jane | Smith | 
| 3 | Jimbo | Jones | 
| 4 | Andy | Smith | 
| 5 | Chris | Jones | 
| 6 | Anna | Bell | 
| 7 | Jimmy | Carr | 
| 8 | Albert | Smith | 
| 9 | John | Doe | 
+--------- +----------- +---------- 十 
+--------- +------------------- 二 
| name_id | email | 
oe PE 二 
| 2 | jsmith@jsmith.com | 
| 6 | annabell@aol.com | 
| 9 | jdoe@yahoo.com | 
Herent sarah ees aa ee + 


在 这 两 个 表 上 使 用 LEFT JOIN， 我 们 可 以 看 到 ， 如 果 email 表 中 的 一 
个 值 不 存在 ， 一 个 空 值 将 会 出 现在 一 个 Email 地 址 的 位 置 ， 执 行 如 下 语 
ile 


SELECT firstname, lastname, email FROM master_name 
LEFT JOIN email ON master_name.name_id = email.name_id; 


这 条 LEFT JOIN 查询 产生 如 下 的 结果 。 


ae, Spare ert TI apes, eve os imete ae ee en eee + 
| firstname | lastname | email | 
fw ae anii fie oe eaa foes ee eee ee ee + 
| John | Smith | | 
| Jane | Smith | Jsmith@jsmith.com | 
| Jimbo | Jones | | 
| Andy | Smith | | 
| Chris | Jones | | 
| Anna | Bell | annabell@aol.com | 
| Jimmy | Carr | | 
| Albert | Smith | | 
| John | Doe | jdoe@yahoo.com | 
OR ee ee EE ae aa ae ES 十 
9 rows in set {0.00 sec) 


一 个 RIGHT JOIN 的 作用 和 LEEFT JOIN 类 似 ， 只 不 过 表 的 顺序 相 
反 。 换 句 话说 ， 当 使 用 RIGHT JOIN 的 时 候 ， 第 二 个 表 中 的 所 有 的 行 都 
将 返回 ， 不 管 它们 在 第 一 个 表 中 是 否 有 匹配 。 然 而 ， 在 master_name 和 
email 表 的 例子 中 ，email 表 中 只 有 3 行 ， 而 master_name 表 中 有 有 9 行 。 这 束 
意味 看 ，master_name 表 中 的 9 行 中 只 有 3 行 会 运 回 ， 执 行 如 下 语句 。 


SELECT firstname, lastname, email FROM master name 
RIGHT JOIN email ON master name.name id = email.name_id; 


25 RFA — IE SR TAN © 


pease teats os es. Rees Sats sn ee Pees e TEE 2 a + 
| firstname | lastname | email | 
es re mealies 人 十 
| Jane | Smith | Jsmith@jsmith.com | 
| Anna | Bell | annabell@aol.com | 
| John | Doe | J}doe@yahoo.com | 
bessai omies ee megi pees Sass E + 


3 rows in set {0.00 sec) 


MySQL 中 有 几 种 不 同类 型 的 JOIN 可 以 使 用 ， 我 们 已 经 学 习 了 最 常 
见 的 类 型 。 要 学 习 CROSS JOIN、STRAIGHT JOIN 和 NATURAL JOIN 这 
样 的 JOIN， 请 全 看 位 于 http://dev.mysql.com/doc/refman/5.5/en/join.html 
的 MySQL 手 册 。 当 你 继续 学 习 的 时 候 ， 我 强烈 建议 你 了 解 并 练习 
JOIN， 这 是 SQL 工 其 箱 中 最 强大 的 一 蒜 工 其 之 一 。 


16.6.2 ”使 用 子 查 询 


简单 地 说 ， 一 个 子 得 询 束 是 出 现在 万 一 条 SQL 语句 中 的 一 条 
SELECT 语句 。 这 样 的 一 个 查询 很 有 用 ， 因 为 它们 往往 去 除了 大 量 的 
JOIN 得 询 的 需要 。 在 应 用 程序 编程 的 例子 中 ， 子 碍 询 可 以 避免 了 在 循环 
中 进行 多 个 查询 的 必要 。 

基本 于 查询 语法 的 一 个 例子 如 下 所 示 。 


SELECT expressions and columns FROM table name WHERE somecolumn = (SUBQUERY}; 


我 们 还 可 以 用 带 有 UPDATE 和 DELETE 语 句 的 子 查询 ， 如 下 所 示 。 


DELETE FROM table name WHERE somecolumn = {SUBQUERY}; 


或 


UPDATE table name SET somecolumn = ‘something’ WHERE somecolumn = {SUBQUERY}; 


提示 : 





一 个 子 查 询 的 外 围 语句 可 以 是 SELECT、INSERT、UPDATE、DELETE、SET 或 DO。 子 
丛 询 必须 总 是 出 现在 括号 中 ， 没 有 例外 。 


当 我 们 使 用 子 查询 的 时 候 ， 外 部 语句 的 WHERE 部 分 不 一 定 必 须 使 


用 = 比较 操作 符 。 除 了 =， 我 们 可 以 使 用 任何 的 基本 比较 操作 符 以 及 IN 这 
样 的 关键 字 。 


如 下 的 例子 使 用 一 个 子 伍 询 来 获取 master_name 表 中 那些 在 email 表 
中 拥有 一 个 Email 地 址 的 用 户 的 记录 。 


SELECT firstname, lastname FROM master_name 
WHERE name_id IN (SELECT name_id FROM email}; 


XR AWN AGAR AY HE FATA o 


十 一 -一 一 一 一 “= 十 一 一 一 一 一 + 
| firstname | Lastname | 
+----------- +---------- 十 
| Jane | Smith | 
| Anna | Bell | 
| John | Doe | 
asst i i a Bele E + 


3 rows in set (0.00 sec) 


要 了 解 关 于 子 租 询 的 更 多 讨论 ， 包 括 限 制 ， 请 参考 位 于 
http://dev.mysql.com/doc/refman/ 5.5/en/subqueries.html 的 MySQL 手 册 的 


Ff ELVA HBP o 


16.7 使 用 UPDATE 命 令 来 修改 记录 


UPDATE 是 用 来 修改 己 有 的 单条 或 多 条 记录 中 的 一 列 或 多 列 的 内 容 
的 SQL 命令 。 最 基本 的 UPDATE 语 法 如 下 所 示 。 


UPDATE table name 

SET columni='new value’, 
column2='new value2' 

[WHERE some condition is true] 


更 新 一 条 记录 有 的 规则 和 插入 一 条 记录 的 规则 类 似 ， 输入 的 数据 必须 
和 字段 的 数据 类 型 对 应 ， 并 且 必 须 用 单 引号 或 双 引 与 把 字符 串 括 起 来 ， 
必要 的 时 候 要 转 义 。 例 如 ， 假 设 你 有 一 个 名 为 fruit 的 表 ， 其 中 包含 一 个 
ID、 一 个 水 果 名 称 和 水 果 的 状态 (ripe 或 rotten) 。 


a SUNY PENE a + 
| 1d | fruit_name | status | 
十 ---- 十 ------------ 十 -------- + 
| 1 | apple | ripe | 
| 2 | orange | rotten | 
| 3 | grape | ripe | 
| 4 | banana | rotten | 
esa Pees aoe 2 Se oases as + 
4 rows in set (0.00 sec) 


要 把 水 果 的 状态 改 为 ripe， 使 用 如 下 语句 。 


UPDATE fruit SET status = ‘ripe'; 


你 将 会 得 到 来 目 数 据 库 的 如 下 一 条 啊 应 。 


Query OK, 2 rows affected (0.00 sec) 
Rows matched: 4 Changed: 2 Warnings: ø 


进一步 看 一 下 这 个 得 询 的 结束 ， 它 成 功 执 行 了 ， 我 们 从 Query OKI 
恩 台 可 以 看 出 来 。 还 要 注意 ， 只 有 两 行 受 到 影响 ， 如 琳 我 们 要 把 一 列 的 


值 设置 为 它 已 经 拥有 的 值 ， 更 新 将 不 会 对 该 列 进 行 。 啊 应 消息 的 第 2 行 
显示 ， 有 4 行 己 经 匹配 了 ， 并 且 只 有 两 行 修改 了 。 如 琳 想 知道 哪些 行 匹 
配 上 了 ， 管 采 很 饭 单 。 因 为 我 们 没有 制定 一 个 特定 的 条 件 进行 匹配 ， 那 
么 所 有 的 行 都 匹配 上 了 。 


在 更 新 一 个 示 的 时 候 ， 我 们 必须 小 心 仔细 并 且 使 用 一 个 条 件 ， 除 非 
真 的 想 把 所 有 记录 的 所 有 列 都 修改 为 相同 的 值 。 为 了 证 明 这 一 点 ， 假 
设 “grape” 在 表 中 没有 拼写 正确 ， 并 且 我 们 想 使 用 UPDATE 来 修改 这 个 错 
误 。 这 个 俘 询 将 会 产生 可 怕 的 后 来 。 


UPDATE fruit SET fruit name = ‘grape’; 


IZA WHA R H REIRA U PZR o 


Query OK, 4 rows affected (0.00 sec) 
Rows matched: 4 Changed: 4 Warnings: @ 


4 ECT RNIB, SB I, ARM, Ñ 
意味 着 fruit 表 内 容 如 下 所 示 ， 


+----t------------ +-------- + 
| id | fruit name | status | 
+----t------------ +-------- + 
| 1 | grape | ripe | 
| 2 | grape | ripe | 
| 3 | grape | ripe | 
| 4 | grape | ripe | 
+----t--------+---- +-------- + 


4 rows in set (0.00 sec) 


现在 ， 所 有 的 水 果 记 录 都 是 grape。 然 而 ， 当 你 试图 更 改 一 条 记录 
的 拼写 的 时 候 ， 所 有 的 记录 都 会 更 改 ， 因 为 你 没有 指定 一 个 条 件 。 在 将 
UPDATE 权 限 赋 了 予 用 户 的 时 候 ， 考 谍 一 下 你 给 予 某 人 的 责任 ， 一 次 错误 
的 更 新 ， 整 个 表 都 将 是 grape。 在 前 面 的 例子 中 ， 你 本 来 应 该 在 WHERE 


子 句 中 使 用 id 或 fruit_ name 字段 ， 正 如 我 们 在 后 面 的 小 节 将 要 看 到 的 。 
16.7.1 条 件 式 UPDATE 


进行 一 次 条 件 式 UPDATE 意 味 着 使 用 WHERE 子 句 来 匹配 特定 记 
录 。 在 UPDATE 语 名 中 使 用 一 个 WHERE 子 句 和 在 SELECT 语句 中 使 用 一 
个 WHERE 子 句 是 相同 鸭 。 所 有 相同 的 比较 操作 符 和 逻辑 操作 符 一 样 可 
以 使 用 ， 例 如 等 于 、 大 于 、OR 和 AND。 


假设 fruit 表 没有 完全 用 grape 盾 充 而 是 包含 4 条 记录 ， 其 中 的 一 条 记 
录 有 一 个 拼写 错误 ， 写 成 了 grappe 而 不 是 grape。 用 来 修改 拼写 错误 的 
UPDATE 语 句 如 下 所 示 。 


UPDATE fruit SET fruit name = ‘grape’ WHERE fruit_name = 'grappe'; 
在 这 个 例子 中 ， 只 有 一 行 罗 配 ， 并 且 只 有 一 行 修改 ， 如 下 和 面 的 结果 
所 示 。 


Query OK, 1 row affected (0.00 sec) 
Rows matched: 1 Changed: 1 Warnings: @ 


我 们 的 fruit 表 应 该 是 完整 的 ， 并 且 ， 所 有 的 水 果 名 应 该 是 拼写 正确 
的 。 


SELECT * FROM fruit; 


这 条 SELECT 查 询 显 示 如 下 结果 。 


人 Speer shea + 
| id | fruit_name | status | 
十 ---- 十 ------------ 十 -------- 十 
| 1 | apple | ripe | 
| 2 | pear | ripe | 
| 3 | banana | ripe | 
| 4 | grape | ripe | 
PAoa SEREEN See DT ny o + 


4 rows in set (0.00 sec} 


16.7.2 ”在 UPDATE 中 使 用 已 有 的 列 值 


UPDATE 的 另 一 个 功能 是 使 用 记录 中 当前 的 值 作为 一 个 基准 值 。 例 
如 ， 回 到 grocery_inventory 表 的 例子 ， 假 设 有 如 下 的 一 张 表 。 


OSs LOS CARS OMe See Aue [es PEPS es eR oe | ae NRE PCIE SIONS, SN 十 
| id | item name | item desc | item price | curr_qty | 
aos eet SD Sp Wa Dee BE DE eee, MA DAADE a Sea + 
| 1 | Apples | Beautiful, ripe apples. | 0.25 | 1800 | 
| 2 | Bunches of Grapes | Seedless grapes. | 2.99 | 500 | 
| 3 | Bottled Water (6-pack}) | 580ml spring water. | sh | 252 | 
| 4 | Bottled Water (12-pack) | 500m1 spring water. | 4.49 | 500 | 
| 5 | Bananas | Bunches, green. | 1.99 | 150 | 
| 6 | Pears | Anjou, nice and sweet. | @.5 | 500 | 
| 7 | Avocado | Large Haas variety. | @.99 | 750 | 
se 二 Bs se i sain 十 


7 rows in set (0.00 sec) 


当 示 个 人 购买 了 一 件 商 品 ， 例 如 一 个 竺 果 〈id=1) ，inventory 表 将 
相应 地 更 新 。 然 而 ， 我 们 不 知道 在 curr_qty 列 中 输入 什么 数值 ， 只 知道 
忌 挥 了 一 件 。 在 这 种 情况 下 ， 使 用 该 列 的 当前 值 并 且 减 1， 语 句 如 下 。 


UPDATE grocery_inventory SET curr_qty = curr gqty - 1 WHERE id = 1; 


这 会 在 curr_qty 列 给 出 一 个 999 的 新 值 ， 实 际 上 正 是 如 此 。 


SELECT * FROM grocery inventory; 


这 条 SELECT 奏 询 显示 了 新 的 库存 数量 ， 如 下 所 示 。 


oem dn in, da Oe de 
| 1 | Apples 

| 2 | Bunches of Grapes 

| 3 | Bottled Water (6-pack}) 

| 4 | Bottled Water (12-pack) 
| 5 | Bananas 

| 6 | Pears 

| 7 | Avocado 
+----+------------------------- 
7 rows in set {0.00 sec) 


Beautiful, ripe apples. 
Seedless grapes. 

500ml spring water. 
500ml spring water. 
Bunches, green. 

Anjou, nice and sweet. 
Large Haas variety. 


16.8 ”使 用 REPLACE 命 令 


修改 记录 的 另 一 个 方法 是 使 用 REPLACE 命 令 ， 它 类 似 于 INSERT 命 
令 。 


REPLACE INTO table name (column list) VALUES (column values}; 


REPLACE 语 句 像 这 样 工 作 : 如 果 插 入 到 表 中 的 记录 包含 了 一 个 主 
键 值 ， 这 个 主键 值 和 表 中 已 有 的 一 条 记录 的 主键 值 相 等 ， 表 中 的 记录 将 
被 删除 掉 ， 并 且 新 的 记录 会 插入 到 它 所 在 的 位 置 。 


提示 : 


REPLACE 命 令 是 MySQL 对 ANSI SQL 的 一 个 特定 的 扩展 。 这 条 命令 模拟 了 DELETE 的 操 
作 并 重新 INSERT 一 条 特定 记录 的 操作 。 换 句 话 说， 我 们 用 一 条 命令 完成 了 两 条 命令 的 工作 。 





使 用 grocery_inventory 表 ， 如 下 的 命令 将 答 换 Apple 的 记录 。 


REPLACE INTO grocery inventory VALUES 
(1, ‘Granny Smith Apples', Sweetl ， '0.50', 1000); 


结果 如 下 所 示 。 


Query OK, 2 rows affected (0.00 sec) 


BW ZR A, ARARAS FY “2 rows affected”。 在 这 个 例子 
中 ， cians ‘E grocery_inventory#< FA MT MANE ‘PS EGE, eH) 
的 行 被 删除 而 一 个 新 行 插入 ， 因 此 是 2 行 受 到 影响 。 使 用 SELECT 语句 碍 
询 数 据 以 验证 记录 是 正确 的 ， 结 果 如 下 。 


Pees tee ers ore SS ree oe yn J Ns + 
| id | item name | item_desc | item_price | curr_qty | 
下 ie Berns Serer sevens a fritid SSAA ES Seats a R E E + 
| 1 | Granny Smith Apples | Sweet! | 0.50 | 1800 | 
| 2 | Bunches of Grapes | Seedless grapes. | 2.99 | 500 | 
| 3 | Bottled Water (6-pack}) | 580ml spring water. | 2.29 | 252 | 
| 4 | Bottled Water (12-pack}) | 500ml spring water. | 4.49 | 500 | 
| 5 | Bananas | Bunches, green. ME | 150 | 
| 6 | Pears | Anjou, nice and sweet. | @.5 | 500 | 
| 7 | Avocado | Large Haas variety. | @.99 | 750 | 
Fenna ies ype epee E a 有 he a se er + 


7 rows in set {0.00 sec) 


gn FG —-REPLACEIS®), Jf Apidae FY EEN AIA MK 
中 已 经 存在 的 一 个 主键 的 值 死 配 ， 这 条 记录 会 直接 搬入， 这 样 就 只 有 一 


条 记录 受到 影 啊 。 


16.9 ”使 用 DELETE 命 令 


DELETE 的 基本 语法 如 下 。 


DELETE FROM table name 
[WHERE some condition is true] 
[LIMIT rows] 


注意 ， 在 DELETE 命 令 中 如 果 没 有 和 条件， 那么 当 你 使 用 DELETE 的 
时 候 ， 表 中 所 有 记录 都 会 被 删除 择 。 你 可 能 还 记得 在 本 音 前 面 的 一 次 关 
于 fruit 表 中 的 grape 的 失败 ， 当 更 新 一 个 表 而 没有 指定 条 件 的 时 候 ， 导 至 
所 有 的 记录 都 更 新 了 。 在 使 用 DELETE 时 候 也 要 小 心 出 现 类 似 的 情况 。 


假设 一 个 fruit 的 表 的 结构 和 数据 如 下 所 示 。 


Crepes Ses E Ee + 
| 1d | fruit name | status | 
Fd ee es Tn i + 
| 1 | apple | ripe | 
| 2 | pear | rotten | 
| 3 | banana | ripe | 
| 4 | grape | rotten | 
+----+------------ +-------- + 


4 rows in set (0.00 sec) 


如 下 的 语句 删除 了 该 表 中 的 所 有 记录 。 
DELETE FROM fruit; 

我 们 总 是 可 以 通过 对 表 使 用 SELECT 语句 来 验证 删除 。 在 删除 所 有 
记录 后 执行 下 面 这 条 命令 。 
SELECT * FROM fruit; 


将 会 看 到 fruit 表 中 所 有 的 水 果 痢 已 经 删除 挥 了 。 


Empty set (0.00 sec) 
条 件 式 DELETE 


一 个 条 件 式 DELETE 语 句 ， 和 条 件 式 SELECT 或 UPDATE 话 人 句 一 
样 ， 意 味 着 我 们 使 用 WHERE 子 句 来 匹配 特定 的 记录 。 我 们 有 了 全 部 可 
用 的 比较 操作 符 和 逻辑 操作 符 ， 因 此 ， 可 以 挑选 和 选取 要 删除 哪些 记 
录 。 


一 个 基本 的 例子 是 ， 从 fruit 表 中 删除 所 有 状态 为 rotten 的 水 果 的 记 
录 ， 如 下 上 所 示 。 


DELETE FROM fruit WHERE status = 'rotten'; 


ESM RPA Aa, TAA AE KAR SS REE BOK 0 


Query OK, 2 rows affected (0.90 sec) 


AWARAU F o 
下 pees. Ree + 
| id | fruit name | status | 
en er sane + 
| 1 | apple | ripe | 
| 3 | banana | ripe | 
BE eee a ta eee i SE erie Se 十 
2 rows in set (0.00 sec} 


我 们 也 可 以 在 DELETE 语 句 中 使 用 ORDER BY 子 句 ， 下 面 看 一 下 把 
ORDER BY 子 句 深 加 到 结构 中 以 后 的 基本 DELETE 语 法 。 


DELETE FROM table_name 

[WHERE some condition is true] 
[ORDER BY some column [ASC | DESC] ] 
[LIMIT rows] 


乍 一 看 ， 你 可 能 会 问 : “为 什么 我 删除 记录 与 顺序 有 关系 呢 >? 


ORDER BY 子 句 不 是 用 来 删除 记录 的 ， 而 是 用 来 排序 记录 的 。 


在 这 个 例子 中 ， 一 个 名 为 access_log 的 表 给 出 了 访问 时 间 和 用 户 
名 ， 如 下 所 示 。 


2012-01-06 06:99:13 
2012-01-06 06:09:22 
2012-01-06 06:09:39 
2012-01-06 06:09:44 


]ohndoe 
janedoe 


4 rows in set (0.00 sec) 


要 删除 最 早 的 记录 ， 首 先 使 用 ORDER BY 来 正确 地 排序 记录 ， 然 
后 ， 使 用 LIMIT 来 仅 删 除 一 条 记录 ， 如 下 上 所 示 。 


DELETE FROM access log ORDER BY date accessed DESC LIMIT 1; 


但 询 access_log 的 记录 ， 从 而 验证 只 有 如 下 3 条 记录 和 存在。 


SELECT * FROM access log; 


结果 如 下 所 示 。 
Se qt hee SS 十 
| id | date accessed | username | 
TE as Senki ree ome iia senate so + 


| 2 | 2012-01-06 06:09:22 | janedoe | 
| 3 | 2012-01-96 06:99:39 | jsmith | 
| 4 | 2012-01-86 06:09:44 | mikew | 
set in rd ee re ee ner pis wanton 十 
3 rows in set (0.00 sec) 


16.10 MySQL? % H AJIT E PR BY 


MySQL 内 建 的 和 字符 串 相 关 的 函数 ， 可 以 以 几 种 方式 使 用 。 我 们 
可 以 在 SELECT 语 句 中 使 用 函数 而 不 用 指定 一 个 表 束 可 以 获取 函数 的 结 
东 。 或 者 我 们 可 以 通过 把 两 个 字段 连接 为 一 个 新 的 字符 串 ， 使 用 函数 来 
扩展 SELECT 的 结 末 。 后 面 的 例子 次 不 是 MySQL 与 字符 串 相 关 的 函数 的 
完整 的 库 。 要 了 解 更 多 信息 ， 请 参考 位 于 
http://dev.mysql.com/doc/refman/5.5/en/ string-functions.html 的 MySQL F 
IT 


16.10.1 长度 和 连接 函数 


长 度 和 连接 函数 主要 用 于 字符 串 的 长 上 度 和 把 字符 串 连 接 起 来 。 与 长 
度 相 关 的 函数 包括 LENGTH(O)、OCTET_LENGTHO、CHAR_LENGTH0O 
和 CHARACTER_LENGTH()， 它 们 所 做 的 事情 实际 上 是 相同 的 ， 即 计 
算 字 符 串 中 的 字符 数 。 


SELECT LENGTH('This is cool!'}); 


结果 如 下 所 示 。 
Ee er DEANE ER + 
| LENGTH('This is cool!') | 
+------------------------- 十 
| 13 | 
Fran Sinanin cantons ogee. aus + 


1 row in set (0.00 sec} 


有 趣 的 事情 从 CONCATO 函 数 开始 ， 这 个 函数 用 来 把 两 个 或 多 个 字 
符 串 连接 起 来 。 


SELECT CONCAT('My', ‘'S', QL ); 


XR BAN ZG ARO RATAN 


es Me rs Ds HE + 
| CONCAT('My', 'S', 'QL') | 
ee ee ae eae eee a + 
| MySQL | 
es ONOS + 


1 row in set (0.00 sec} 


假设 我 们 对 一 个 包含 了 名 字 的 表 使 用 这 个 函数 ， 名 字 划 分 为 
firstname 字 段 和 lastname 字 段 。 我 们 使 用 来 连接 firstname 字 段 
和 ]lastname 字 段 的 值 ， 而 不 是 使 用 两 个 字符 串 。 通 过 连接 字段 ， 减 少 了 
在 应 用 程序 中 得 到 相同 结果 所 需 的 代 但 。 


SELECT CONCAT{(firstname, lastname) FROM master_name; 


如 果 如 下 。 


| JohnSmith | 
| JaneSmith | 
| JimboJones | 
| AndySmith | 
| ChrisJones | 
| AnnaBell | 
| JimmyCarr | 
| AlbertSmith | 
| JohnDoe | 


9 rows in set (0.00 Sec} 





QO RE ee BP 8 HI PSA EIN NE PE EB, AEE SR BAAS SSH. WRK 
这 么 做 了 ，MYySQL 逐 字 地 解析 字符 串 。 在 CONCATO0 的 例子 中 ， 我 们 得 到 如 下 结 





SELECT CONCAT('firstname', ‘lastname’ } FROM master name; 


将 会 得 到 如 下 结果 。 


| CONCAT('firstname', 'lastname') | 


| firstnamelastname | 
| firstnamelastname | 
| firstnamelastname | 
| firstnamelastname | 
| firstnamelastname | 
| firstnamelastname | 
| firstnamelastname | 
| firstnamelastname | 
| firstnamelastname | 


9 rows in set (0.00 sec) 





OR A FZ) RAPA od BAT, CONCATQO RI BCH IRA H, 
这 也 就 引出 了 下 一 个 函数 。 


你 可 能 还 记得 ，CONCAT_WS0O 代 表 使 用 分 隔 符 的 连接 。 可 以 选择 
任何 分 隔 符 ， 下 面 的 例子 使 用 空格 。 


SELECT CONCAT WS(' ', firstname, lastname} FROM master_name; 


XR BAN ZG ARB BATA 


| John Smith | 
| Jane Smith | 
| Jimbo Jones | 
| Andy Smith | 
| Chris Jones | 
| Anna Bell | 
| Jimmy Carr | 
| Albert Smith | 
| John Doe | 


9 rows in set (0.00 sec) 


Oy FR AB Ss a Fe ZS TERE, BY OBA ASR A Be aR 
段 ， 示 例如 下 。 


SELECT CONCAT WS(' ', firstname, lastname) AS fullname FROM master_name; 


使 用 这 条 语句 ， 将 会 得 到 如 下 结 来 。 


| fullname | 
| John Smith 

| Jane Smith | 
| Jimbo Jones | 
| Andy Smith | 
| Chris Jones | 
| Anna Bell | 
| Jimmy Garr | 
| Albert Smith | 
| John Doe | 


9 rows in set (0.00 sec) 


16.10.2 截断 和 填充 函数 


MySQL 提 供 了 几 个 函数 来 癌 字 人 符 串 中 添加 和 删除 额外 的 字符 ， 包 
括 空 格 。RTRIMO 和 LTRIMO 函 数 从 一 个 字符 串 的 右 端 或 者 左 端 加 除 空 
格 。 


SELECT RTRIM('stringstring ‘D5 


该 得 询 的 结束 如 下 所 示 ， 尽 管 很 难看 出 它 的 变化 。 


Pee. PES ME, SS meee sens + 
| RTRIM{ 'stringstring Y 
faccina ee ee eee eee + 
| stringstring | 
ise ee cuteeaye ete + 


1 row in set (0.00 sec} 


LTRIMO AUS HA DAA, ANSP. 


SELECT LTRIM(' stringstring'}); 


这 条 查询 导致 如 下 结果 ， 空 白明 显 被 去 掉 了 ， 示 例如 下 。 


Site) eee ve Sar, Deere See 十 
| LTRIM(' stringstring') | 
to + 
| stringstring | 
十 ------------------------- 十 


1 row in set (0.00 sec} 


如 采 字 人 符 串 产生 目 一 个 固定 宽度 的 字段 ， 并 且 它 要 么 不 需要 添加 其 
他 的 填充 ， 要 么 将 插入 到 一 个 varchar 或 其 他 非 固定 宽度 的 字段 ， 我 们 可 
以 规 断 捧 宛 过 的 字符 串 。 如 采 字 符 串 使 用 空格 以 外 的 其 他 字符 来 填充 ， 
使 用 TRIMO 函 数 来 指定 想 要 删除 的 字符 。 例 如 ， 要 把 前 面 的 X 字 人 符 从 字 
侍 串 XXXneedleXXX 中 删除 ， 使 用 如 下 语句 。 


SELECT TRIM(LEADING 'X' FROM 'XXXneedlexXxx' }; 


IX A A ARB o 


a e rs eevee a HERE Ane RE Seee Sta es tomes, sane + 
| TRIM{LEADING 'X' FROM 'XXXneedleXxxx') | 
hn eee + 
| need1lexxx | 
dol nd ature Sushma. cub a Wiehe ee a ee + 


1 row in set {0.00 sec} 


可 以 使 用 TRAILING 从 字符 串 末 尾 删 除 字 符 。 


SELECT TRIM{TRAILING 'X' FROM 'XXXneedlexxXx' }; 


XK AAG ARG FZR o 


人 十 
| TRIM{TRAILING 'X' FROM 'XXXneedleXXX') | 
+---------------------------------------- 十 
| XXXneedle | 
ria O E a a i i + 


1 row in set (0.00 sec) 


如 果 没 有 指定 LEADING 或 TRAILING， 则 假设 两 种 方式 都 使 用 ， 语 
AJU F o 


SELECT TRIM('X' FROM 'XXXneedleXXX')}; 


这 条 查询 结果 如 下 。 


er, Sh Ss. oi SH oes, Bites + 
| TRIM('X' FROM 'XXXneedlexxx') | 
二 + 
| needle | 
十 ------------------------------- 十 


1 row in set {0.00 sec} 


和 RTRIMO 和 LTRIMO 删 除 填充 字符 一 样 ，RPADO 和 LPADO 向 一 
个 字符 串 添 加 字符 。 例 如 ， 在 用 于 销售 的 一 个 数据 库 中 ， 对 于 作为 订单 
号 的 一 部 分 的 一 个 字符 串 ， 我 们 可 能 想 要 添加 一 个 特定 的 标识 字符 。 当 
我 们 使 用 填充 函数 时 ， 所 需 的 元 每 是 字符 串 、 目 标 长 度 以 及 填充 字 人 和 从。 
例如 ， 用 一 个 X 字 符 填 充 字 符 串 needle， 直 到 字符 串 达 到 10 个 字符 的 长 
度 ， 使 用 如 下 语句 。 


SELECT RPAD('needle', 10, ‘X'}; 


结 采 如 下 。 
eS Peas Rte, eS eek 十 
| RPAD{'needle', 10, 'X') | 
A or crema S. stu + 
| needlexxxXxX | 
aab ER RE 和 munis Bits 十 


1 row in set (0.00 sec} 


16.10.3 ”定位 和 位 置 函数 


定位 和 位 置 函 数 用 来 在 万 一 个 字符 串 中 得 找 一 个 字符 串 的 部 分 。 
LOCATEO 函 数 返 回 一 个 给 定 的 子 字 符 串 在 目标 字符 哩 中 第 一 次 出 现 的 


人 位置。 例如， 我 们 可 以 在 一 个 haystack 中 查找 needle， 语 名 如下。 


SELECT LOCATE('needle’', ‘haystackneedlehaystack' }; 


应 该 会 看 到 如 下 结果 。 


a ee a SURO HR A kate + 
| LOCATE('needle', ‘haystackneedlehaystack') | 
人 十 
| 9 | 
tee eee ee ee ee ee ee ee ee ee ee eee ee ee ee ee eee + 


1 row in set {0.00 sec) 


子 字符 串 needle 从 目标 字符 串 的 第 9 个 字符 开始 。 如 果 没 有 在 目标 字 
符 串 中 找到 子 字符 串 ，MySQL 返 回 0 作 为 结果 。 


fie 7s: 





和 大 多 数 程序 设计 语言 中 的 位 置 计 算 不 同 ， 那 些 语言 中 的 位 置 计 算 都 是 从 0 开始 的 ， 而 
MySQL 则 是 从 1 开始 计算 位 置 的 。 





LOCATE() 函 数 的 一 个 扩展 是 对 起 始 位 置 使 用 第 3 个 参数 。 如 果 从 
haystack 中 的 位 置 9 之 前 开始 查找 needle， 我 们 将 能 够 得 到 一 个 9 的 结果 。 
含 则 ， 由 于 needle 是 从 位 置 9 开始 的 ， 如 末 我 们 指定 了 一 个 更 大 的 数字 作 
为 开始 位 置 ， 将 会 得 到 一 个 0 的 结果 。 


16.10.4 FF pK ay 


从 一 个 目标 字符 串 中 提取 一 个 子 字符 串 ， 有 几 个 函数 能 够 满足 要 
求 。 给 定 一 个 字符 串 、 起 始 位 置 和 长 度 ， 我 们 可 以 使 用 SUBSTRING0O 
函数 。 如 下 例子 从 字符 串 MySQL 中 获取 3 个 字符 ， 从 位 置 2 开始 。 


SELECT SUBSTRING{ "MySQL", 2, 3); 


结果 如 下 所 示 。 


Pema. See, Sete eS aes + 
| SUBSTRING("MySQL", 2, 3) | 
ee ae a eae ee + 
| ysa | 
i 十 


1 row in set (8.00 sec} 
GUAR JR REE PRE A An mA vig J LEE, AY A AA LEFT) 7 
RIGHTOR2, aw F o 


SELECT LEFT( "MySQL", 2); 


这 条 奏 询 的 结束 如 下 。 


poeemi renot Zr rak + 
| LEFT("MySQL", 2) | 
Ts SANE ee IINR e eats + 
| My | 
Pedro aeti mee. eS $ 


1 row in set (0.00 sec) 


同样 的 ， 使 用 RIGHTO 函 数 示 例如 下 。 


SELECT RIGHTI( MYSQL , 3); 


这 条 奏 询 的 结束 如 下 。 


站 十 
| RIGHT("MySQL", 3) | 
sete st yer aren aad mech saad + 
| SQL | 
Fee ee ee ee + 


1 row in set (0.00 sec) 


YP PAE EB PRIA & YD IE Ze SE E A A — BaP RG 
看 是 谁 下 了 这 个 订单 。 在 一 些 应 用 程序 中 ， 系 统 设 计 来 产生 一 个 包含 日 
SH, Se Papa Ae A StS. UR PT BS Ae eR 
特定 的 模式 ， 如 XXXX-YYYYY-ZZ， 我 们 可 以 使 用 子 字符 串 函 数 来 提 


取 整 个 订单 号 码 的 单个 部 分 。 人 例如， 如果 ZZ 总 是 表示 订单 要 发 往 哪个 
州 ， 我 们 可 以 使 用 RIGHTO 函 数 来 提取 这 些 字 符 并 且 显 示 订 单 中 关于 送 
往 哪 个 州 的 数字 。 
16.10.5 ZIF E1 MA A 

我 们 所 选择 的 程序 设计 语言 可 能 有 修改 字符 串 的 函数 ， 但 是 ， 如 时 
可 以 执行 任务 作为 SQL 语 句 的 一 部 分 ， 那 会 更 好 ， 尺 可 能 地 让 数据 库 系 
统 做 更 多 的 工作 。 


MySQL 的 LCASEO 和 UCASE() 函 数 把 一 个 字符 串 转 换 为 小 写 的 或 大 
写 的 ， 示 例如 下 。 


SELECT LCASE({ 了 YSGL ) 


XR AVATAR F 


TOS A eee es ote + 
| LCASE('MYSQL') | 
全 二 enerenceiat 十 
| mysql | 
上 十 


1 row in set (@.08@ sec) 


要 将 其 变 成 大 与 字符 ， 使 用 如 下 语句 。 


SELECT UCASE({'mysql'); 


IR BAS ae Fe BO BAG KR 


站 十 
| UCASE('mysql') | 
i DAAA + 
| MYSQL | 
----------------- 十 


1 row in set (0.00 sec) 


提示 : 





当 我 们 根据 存储 在 MySQL 中 的 数据 验证 用 户 输入 的 时 候 ， 例 如 ， 在 用 户 登 录 表 单 的 情况 
下 ，LCASE() 和 UCASE() 函 数 很 实用 。 如 果 我 们 想 要 登录 过 程 不 区 分 大 小 写 ， 可 以 尝试 用 用 户 
输入 的 大 写 (或 小 写 ) 版 本 来 匹配 存储 在 表 中 的 数据 的 大 写 (或 小 写 ) 版 本 。 





记 住 ， 如 条 把 字段 名 和 函数 一 起 使 用 ， 不 要 使 用 引 扎 ， 示 例如 下 。 


SELECT UCASE(lastname) FROM master_name; 


EA EA ee, Kattan p 


| 
| 
| 
| 
| 
| 
SMITH | 
| 
| 
+ 


9 rows in set (0.00 sec) 


另 一 个 有 趣 的 字符 串 操 作 函 数 是 REPEATO， 它 所 做 的 事情 正如 其 
名 字 那 样 ， 重 复 一 个 字符 串 给 定 的 次 数 ， 示 例如 下 。 


SELECT REPEAT({"bowwow", 4); 


应 该 会 看 到 如 下 的 结 末 。 


ae oe ae a iei + 
| REPEAT("bowwow", 4) | 
pane ee ee ee eee eee + 
| bowwowbowwowbowwowbowwow | 
ns + 


1 row in set (0.00 sec} 


REPLACE() chi Ss S BN FF BB PS 29 EB A Sa 
换 挥 ， 示 例如 下 。 


SELECT REPLACE( ‘bowwowbowwowbowwowbowwow', ‘wow’, WOW'}; 


OR BVA ae" EU PAG 


yr 十 
| REPLACE ( ' bowwowbowwowbowwowbowwow ‘wow, WoW ) | 
a ne ee ee ee a ee ee ee + 
| bowWOWbowWOWbowWOWbowWwOW | 
ee 十 


1 row in set {@.00 sec) 


16.11 *& MySQL? (4% H H HA AEN Tal eK ži 


MySQL W E HIA FSA SC EN RK a FY DA A FE SELECT 18 8) OR ARM eK 
BUNA, THERA TE DRAN. RA, AY DASE EA SS AE EY H H 
字段 〈 如 日 期 、 日 期 时 间 、 时 间 戳 、 年 份 等 ) 和 函数 一 起 使 用 。 


根据 所 用 的 字段 类 型 ， 和 日 期 相关 的 函数 的 结果 或 多 或 少 部 有 用 
处 。 下 面 的 例子 绝 不 是 MySQL 日 期 和 时 间隙 数 的 完整 库 。 要 了 解 更 多 
言 轧 ， 请 参阅 位 于 http:Wdev.mysql.comy/doc/ refman/5.0/en/date-and-time- 
functions.html 的 MySQL 手 册 。 


16.11.1 操作 日 期 


DAYOFWEEK0O 和 WEEKDAY0 函 数 做 类 似 的 事情 ， 但 是 结果 略 有 
不 同 。 这 两 个 冰 数 都 用 来 查找 一 个 日 期 的 星期 过 引 ， 但 是 ， 不 同 之 处 在 
于 开始 日 期 和 位 置 。 


如 果 使 用 DAYOFWEEK()， 一 个 星期 的 第 一 天 是 星期 日 ， 位 置 为 
1。 一 个 星期 的 最 后 一 天 是 星期 六 ， 位 置 为 7， 示 例如 下 。 


SELECT DAYOFWEEK{ 2012-01-09 ) 


ARAW EUW TH 


er rar re te h ee + 
| DAYOFWEEK('2012-01-@9') | 
下 is 十 
| 2 | 
十 ------------------------- 十 


1 row in set (0.00 sec} 


结果 显示 ，2006 年 1 月 9 日 的 星期 索引 是 2， 也 束 是 说 该 天 是 星期 
。 对 于 WEEKDAY0 函 数 使 用 同样 的 日 期 ， 将 会 得 到 不 同 的 结果 ， 但 
含义 是 相同 的 ， 示 例如 下 。 


信义 

es ep er bedea + 
| WEEKDAY({'2012-01-09') | 
E A E E E TE + 
| 0 | 
a - 十 


1 row in set (0.00 sec} 


结果 显示 ，2006 年 1 月 9 日 的 星期 索引 是 0。 因 为 WEEKDAY0O 使 用 
星期 一 作为 星期 的 第 一 天 ， 位 置 为 0， 而 使 用 星期 日 作为 最 后 一 天 ， 位 
置 为 6。 结 果 为 0 是 正确 的 ， 表 示 这 一 天 是 星期 一 。 


DAYOFMONTHO 和 DAYOFYEARO 函 数 更 加 直接 ， 只 有 一 个 结 
K, FFA, DAYOFMONTHOH 24 RHE M131, MDAYOFYEAR() 
的 范围 从 1 到 366。 看 下 面 的 例子 。 


SELECT DAYOFMONTH( '2012-01-09'); 


这 条 查询 产生 如 下 的 结果 。 


ee oe a aE we + 
| DAYOFMONTH{ '2012-01-09') | 
i + 
| 9 | 
T0000000000000000000000000 十 


1 row in set (0.00 sec} 


现在 试 试 下 面 的 例子 。 


SELECT DAYOFYEAR{ 2012-01-09 }; 


这 条 查询 产生 如 下 的 结果 。 


1 row in set (0.00 sec} 

根据 一 个 特定 的 日 期 来 返回 它 在 月 份 中 的 天 数 ， 看 上 去 似乎 有 些 奇 
怪 ， 因 为 这 个 天 数 就 在 日 期 字符 串 中 。 但 是 ， 考 虑 一 下 在 WHERE 子 句 
中 使 用 这 一 函数 来 对 记录 进行 比较 。 如 果 我 们 有 一 个 表 保 存 了 在 线 订 
单 ， 其 中 一 个 字段 包含 了 下 订单 的 日 期 我 们 可 以 很 快 地 得 到 一 周 中 任 
何 一 天 的 订单 数量 ， 或 者 看 到 前 半 个 月 和 后 半 个 月 分 别 有 多 少 订单 。 

如 下 的 两 个 查询 显示 了 在 每 个 星期 的 前 三 天 中 《包括 各 个 月 ) 以 及 
这 周 其 他 的 各 天 中 的 订单 数 。 
SELECT COUNT(id) FROM orders WHERE DAYOFWEEK(date ordered) < 4; 
SELECT COUNT (id) FROM orders WHERE DAYOFWEEK(date_ ordered) > 3; 

使 用 DAYOFMONTHO， 如 下 的 例子 给 出 了 在 任何 一 个 月 份 中 前 半 
个 月 和 后 半月 的 订单 数目 。 
SELECT COUNT(id) FROM orders WHERE DAYOFMONTH(date ordered) < 16; 
SELECT COUNT(id) FROM orders WHERE DAYOFMONTH(date_ ordered) > 15; 

可 以 使 用 DAYNAME() 函 数 来 为 结果 添加 更 多 的 生机 ， 因 为 它 会 返 
回 给 定 日 期 的 星期 名 。 


SELECT DAYNAME({date ordered) FROM orders; 


这 条 得 询 产 生 如 下 的 结 琳 。 


| Thursday | 
| Monday | 
| Thursday | 
| Thursday | 
| Wednesday | 
| Thursday | 
| Sunday | 
| Sunday | 


8 rows in set (09.00 sec) 


图 数 并 不 仅 限 于 用 在 WHERE 子 句 中 ， 也 可 以 在 ORDER BY 子 句 中 
使 用 它们 ， 示 例如 下 。 


SELECT DAYNAME {date ordered})} FROM orders ORDER BY DAYOFWEEK(date ordered}; 


16.11.2 ”操作 月 份 和 年 份 


星期 几 并 不 是 日 历 的 唯一 部 分 ，MySQL 也 有 专门 用 于 月 份 和 年 份 
的 函数 。 就 像 DAYOFWEEKO 和 DAYNAMEO0O 函 数 一 样 ，MONTHO 和 
MONTHNAMEO0O 函 数 返 回 了 一 年 中 的 月 份 数 和 给 定 日 期 的 月 份 的 名 
字 ， 不 例如 下 。 


SELECT MONTH{ '2012-01-09'), MONTHNAME({ '2012-01-09'); 


这 条 得 询 产 生 如 下 的 结 琳 。 


a spa) Se See Se DA g a + 
| MONTH('2@12-01-99') | MONTHNAME( '2012-01-09') | 
+--------------------- +------------------------- 十 
| 1 | January | 
十 --------------------- 十 ------------------------- 十 


1 row in set {0.00 sec) 


orders 表 使 用 MONTHNAME() 来 显示 相应 的 结果 ， 但 很 多 都 是 重复 
的 数据 ， 示 例如 下 。 


| November | 
| November | 
| November | 
| November | 
| November | 
| November | 
| November | 
| October | 


8 rows in set (0.00 sec) 


可 以 使 用 DISTINCT 来 获取 非 重复 性 的 结果 ， 示 例如 下 。 


SELECT DISTINCT MONTHNAME {date ordered)} FROM orders; 


ARAWE TER. 


下 十 
| MONTHNAME(date_ordered) | 
Se eee eee + 
| November | 
| October | 
Bia ad Cri yeas Ege ee 1 


2 rows in set (0.00 sec} 


要 操作 年 份 ，YEAR0O 函 数 返回 给 定 日 期 的 年 份 ， 示 例如 下 。 


SELECT DISTINCT YEAR(date ordered) FROM orders; 


ARAWE BA 


RE MEIN GENO D + 
| YEAR(date_ ordered) | 
e ee ee + 
| 2011 | 
| 2012 | 
De eS sips cote eee + 


1 row in set (@.08 sec) 


16.11.3 ”操作 局 


操作 周 可 能 是 有 些 技巧 性 的 地 方 ， 如 果 星 期 日 是 一 周 的 第 一 天 而 12 
月 份 还 没有 到 达 一 周 的 结束 ， 那 么 ， 一 年 中 就 会 有 53 周 。 例 如 ，2001 年 
的 12 月 30 扎 是 星期 日 。 


SELECT DAYNAME( '2001-12-30'); 


这 条 碍 询 产 生 的 结束 如 下 。 


el re E E + 
| DAYNAME{ '2001-12-30') | 
+----------------------- 十 
| Sunday | 
Taudin si ee averse ater + 


1 row in set (0.00 sec} 
这 一 事实 使 得 2001 年 的 12 月 30 号 是 该 年 份 的 第 53 周 的 一 部 分 。 
SELECT WEEK( '2001-12-30' }; 


结 柴 中 显示 出 了 该 年 份 所 拥有 的 正 硝 的 星期 效 。 


并 十 
| WEEK('2001-12-30') | 
Sic elites cs ag eR Se Rea uta ates + 
| 53 | 
Feios e ERa ee EE + 


1 row in set (0.00 sec) 


第 53 周 包含 12 月 30 日 和 12 月 31 日 ， 只 有 两 天 ，2002 年 的 第 一 周 从 1 
月 1 日 开始 。 


如 果 我 们 想 要 一 个 周 从 星期 一 开始 但 仍然 能 够 得 到 一 个 年 份 中 的 星 
期 数 ， 可 选 的 第 二 个 参数 使 得 我 们 能 够 更 改 开始 日 期 。1 表 示 这 个 周 从 
星期 一 开始 。 在 后 面 的 例子 中 ， 从 星期 一 开始 使 得 12 月 30 日 成 为 2001 年 
第 52 周 的 一 部 分 ， 但 12 月 31 日 仍然 是 2001 年 第 53 周 的 一 部 分 。 


SELECT WEEK( '2001-12-30',1}); 


这 条 得 询 产 生 如 下 的 结 琳 。 


ONHE, ESNE, EENE AEE, + 
| WEEK('2001-12-30',1) | 
中 十 
| 52 | 
decease el See om ta + 


1 row in set (0.00 sec) 


而 如 下 的 三 询 。 


SELECT WEEK( '2001-12-31',1); 


产生 如 下 的 结果 。 


1 row in set (0.00 sec) 


16.11.4 操作 小 时 、 分 钟 和 各 


如 果 我 们 想 使 用 一 个 包含 了 确切 时 间 的 日 期 ， 例 如 日 期 时 间或 时 间 
惟 ， 或 者 仅仅 是 一 个 时 间 字 段 ， 也 有 函数 能 够 从 字符 串 中 得 出 小 时 、 分 
钟 和 秒 。 这 并 不 令 人 居 讶 ， 这 些 函 数 叫 做 HOURO、MINUTEO 和 
SECONDO。HOURO 返 回 了 给 定时 间 的 小 时 ， 这 个 值 在 0 到 23 之 间 。 
MINUTEO 和 SECOND0O 的 范围 在 0 到 59 之 间 。 


示例 如 下 。 


SELECT HOUR('2012-01-09 07:27:49') AS hour, 
MINUTE{ '2012-01-@9 07:27:49') AS minute, 
SECOND({'2012-01-@9 07:27:49') AS second; 


eR Er AE UT BY ZAR 


juen y poti sse st spre, gute ores k + 
| hour | minute | second | 
a Paro tite di FORDA + 
| 7 | 27 | 49 | 
ee cme ates ae fone: secsans:s + 
1 row in set (@.00 sec) 


有 很 多 的 查询 可 以 从 一 个 日 期 时 间 字 段 获 取 一 个 时 间 ， 我 们 可 以 把 
小 时 和 分 钟 放 在 一 起 ， 甚 至 可 以 使 用 CONCAT_WSO0O 在 结果 之 间 放 置 一 
个 “:”， 从 而 得 到 一 个 时 间 的 表示 ， 示 例如 下 。 


SELECT CONCAT_WS(':',HOUR('2012-01-09 07:27:49'), 
MINUTE{ '2012-01-09 07:27:49')) AS sample time; 


ARAW AE UN BY GR 


PODESI LSE, BES + 
| sample time | 
站 十 
| 7:27 | 
由 十 


1 row in set (0.00 sec) 


16.11.5 ”使 用 MySQL 格 式 化 日 期 和 时 间 


DATE_FORMATO0O 函 数 把 一 个 日 期 、 日 期 时 间或 者 时 间 惟 字段 格式 
化 为 一 个 字符 串 ， 它 通过 选项 来 明确 制定 如 何 显 示 结 果 。 
DATE_FORMAT0 的 语法 如 下 。 


DATE FORMAT(date,format) 


表 16-2 列 出 了 很 多 格式 化 选项 。 


表 16-2 。 ”DATE_FORMATO(O 格 式 化 字符 串 选 项 





%M 月 份 名 称 (January 到 December) 


缩 略 的 月 份 名 称 (Jan 到 Dec) 

之 有 填充 数字 的 月 份 (01 到 12) 

月 份 (1 到 12) 

星期 几 的 名 称 (“Sunday 到 Saturday) 

缩 略 的 星期 几 的 名 称 “Sun 到 Sat) 
mei 如 first、second、third 等 等 
帘 有 填充 的 月 份 中 的 第 几 天 (00 用 31) 
Fine Cost 

THA FASO tN BLA (00121366) 

四 位 数 的 年 份 

两 位 数 的 年 份 

星期 日 是 第 一 天 的 四 位 年 份 ， 和 %V 一 起 使 用 


%x 星期 一 是 第 一 天 的 四 位 年 份 ， 和 %V 一 起 使 用 























poem (0=Sunday, ..., 6=Saturday ) 

星期 日 是 第 一 天 的 星期 数 〈0 到 53 ) 

星期 一 是 第 一 天 的 星期 数 〈0 到 53 ) 

一 天 的 星期 数 〈1 到 53) ， 和 9%X 一 起 使 用 
和 aa (1253) ， 和 %x 一 起 使 用 
帘 有 填充 位 的 小 时 (00 到 23) 

帘 有 填充 位 的 小 时 (01 到 12) 

小 时 (1 到 12) 

市 有 填充 位 的 分 钟 00 到 59) 

帘 有 填充 位 的 秒 数 (00 到 59) 

帘 有 填充 位 的 秒 数 (00 到 59) 

12 小 时 时 钟 的 时 间 (hh:mm:ss [AP]M) 












































%T 247 Ep ENT 24 AAJ ENT E] Chh:mmiss ) 





提示 : 


DATE_FORMAT0 字 符 串 选项 中 使 用 的 任何 其 他 的 字符 都 是 依次 出 现 的 。 


要 显示 我 们 上 一 小 节 中 提 到 的 02:02 的 结果 ， 应 该 使 用 %h 和 %6i 选 项 
来 从 日 期 返回 小 时 和 分 钟 ， 并 且 在 两 个 选项 之 间 带 有 一 个 “:”， 示 例如 
Be 


SELECT DATE_FORMAT('2012-01-09 02:02:00', '%h:%i') AS sample time; 


Ike ET A AE OB A ZR 


Wy SN + 
| sample time | 
a BA + 
| 02:02 | 
nae + 


1 row in set {0.00 sec) 


下 面 只 是 几 个 关于 使 用 DATE_FORMATO0O 函 数 的 例子 ， 但 是 ， 自 己 
练习 使 用 这 个 函数 才 是 理解 它 的 最 好 办 法 。 


SELECT DATE_FORMAT('2012-01-09', ‘SW, %M %D, %Y') AS sample time; 


这 条 得 询 产 生 如 下 的 输出 。 


1 row in set {0.00 sec) 
如 下 是 格式 化 当前 时 间 的 一 条 但 询 ( 对 了 ，NOW0O 表 示 的 是 我 编写 
这 条 语句 时 候 的 时 间 ) 。 


SELECT DATE_FORMAT(NOW(),'%W the %D of %M, %Y 
around 1l o\'clock %p') AS sample time; 


这 条 得 询 产 生 如 下 的 输出 。 


re er Se Se + 
| sample time | 
a EAST A E AADIS pete SEE ER + 
| Tuesday the 18th of January, 2012 around 8 o'clock PM | 
PE E E E E E E + 


1 row in set (0.04 sec} 


民 完 备 ， 并 且 ， 你 会 友 


Se 
一 个 


化 些 时 间 自 己 研究 日 期 格式 化 选项 ， 其 功 
现 它们 很 容易 使 用 。 


16.11.6 ”使 用 MySQL 执 行 日 期 算术 


MVySQL 有 几 个 函数 可 以 用 来 执行 日 期 算术 ， 这 可 能 是 MySQL 做 计 
算 比 使 用 PHP 脚 本 计算 更 快 的 领域 之 一 。 给 定 一 个 起 始 日 期 和 一 个 则 
隔 ，DATE_ADD0O 和 DATE_SUB0O 函 数 返 回 结果 。 两 个 函数 的 语法 分 别 
如 下 。 


DATE ADD(date,INTERVAL value type) 


DATE SUB(date, INTERVAL value type) 


4016-35 7N J PJ REAR ETS Fe AY eg I o 


4216-3 日 期 算术 的 值 和 类 型 


“小 时 : 分 钟 : 秒 数 ” HOUR_SECOND 










“日 期 小 时 : 分 钟 ” DAY MINUTE 


“日 期 小 时 : 分 钟 : 秒 数 ” DAY SECOND 





例如 ， 要 得 到 当前 日 期 加 21 天 的 结果 ， 使 用 如 下 方法 。 


SELECT DATE_ADD(NOW{), INTERVAL 21 DAY}; 


这 条 得 询 产 生 如 下 的 结 琳 。 


NES SHON, AURA MNOS SNe Sete OS, È + 
| DATE_ADD{NOW(), INTERVAL 21 DAY) | 
由 十 
| 2012-01-31 21:02:16 | 
二 -0 十 


1 row in set (@.02 sec) 


使 用 DATE_SUBO 产 生 如 下 的 结果 。 


本 十 
| DATE SUB{NOW(), INTERVAL 21 DAY) | 
a EAT a cei arene ceaeemses 4 + 
| 2011-12-20 21:02:23 | 
-0 + 


1 row in set (@.00 sec) 
(FA OE 16-3PT RREA DEA RIEACITA, ADAYA 
是 DAYS。 使 用 DAYS 会 导致 一 个 如 下 的 错误 。 


ERROR 1064: You have an error in your SQL syntax near DAYS) at Line 1 


如 果 我 们 对 一 个 日 期 值 而 不 是 日 期 时 间 值 使 用 DATE_ADD0 或 
DATE_SUBO0O 函 数 ， 结 采 将 会 显示 为 一 个 日 期 值 ， 除 非 你 使 用 了 关系 到 
小 时 、 分 钟 和 秒 钟 的 表达 式 。 在 那 种 情况 下 ， 结 果 将 是 一 个 日 期 时 间 。 


例如 ， 第 一 个 查询 的 结束 仍然 为 一 个 日 期 字段 ， 而 第 二 个 得 询 的 结 
朱 变 成 了 一 个 日 期 时 间 。 


SELECT DATE ADD("2011-12-31", INTERVAL 1 DAY}; 


IX RET A AE TER 


SPLATT RADNER SOND SURO TEN AeneA A Nee nae 2 + 
| DATE ADD("2011-12-31", INTERVAL 1 DAY) | 
detains PETNE SRNR p DDR nS UE RAT aki, < + 
| 2012-01-01 | 
十 --------------------------------------- - 十 


1 row in set (0.00 sec) 


而 执行 如 下 这 条 碍 询 。 


SELECT DATE _ADD("2011-12-31", INTERVAL 12 HOUR}; 


产生 如 下 的 结果 。 
TE SE PO ES Me Soest + 
| DATE_ADD("2011-12-31", INTERVAL 12 HOUR) | 
F-20000 + 
| 2011-12-31 12:00:80 | 
二 S a a S + 


1 row in set (0.00 sec} 


也 可 以 使 用 + 或 -操作 符 ， 而 不 是 DATE_ADD0O 和 DATE_SUBO 函 数 
来 执行 日 期 算术 ， 示 例如 下 。 


SELECT "2011-12-31" + INTERVAL 1 DAY; 


ARAW EUW TH 


ETRE, ens. ES PETRAS OUND .pe + 
| "2011-12-31" + INTERVAL 1 DAY | 
下 十 
| 2012-01-01 | 
十 ------------------------------- 十 


1 row in set {0.00 sec) 


16.11.7 ”特殊 函数 和 转换 函数 
MySQL 的 NOW0O 因 数 返回 一 个 当前 日 期 时 间 结 果 ， 并 且 对 于 时 间 


稚 登 录 或 访问 时 间 以 及 很 多 其 他 的 任务 都 很 有 用 。MySQL 还 有 一 些 其 
他 的 函数 可 以 执行 闫 似 的 任务 。 


CURDATE() 和 CURRENT_DATE() 函 数 是 相同 的 ， 并 且 它 们 都 以 
YYYY-MM-DD 格 式 返 回 当前 日 期 ， 示 例如 下 。 


SELECT CURDATE(), CURRENT DATE()}); 


Xe ET A AE OB A ZR 


站 gow cheers cee eres 2 + 
| CURDATE() | CURRENT DATE() | 
下 Secures. ee 站 十 
| 2012-01-10 | 2012-01-10 | 
teen eee eee eee 十 ---------------- 十 


1 row in set {0.01 sec) 


类 似 地 ，CURTIME0 和 CURRENT_TIMEO 函 数 以 HH:MM:SS 格 式 
返回 当前 时 间 ， 示 例如 下 。 


SELECT CURTIME(), CURRENT TIME  ) 


这 条 得 询 产 生 如 下 的 结 琳 。 


a AMENE È Rs 十 
| CURTIME(} | CURRENT _ TIME() | 
于 下 十 
| 09:14:26 | 09:14:26 | 
+-------+---- 十 ---------------- 十 


1 row in set {0.00 sec) 


NOW(). SYSDATE()#ICURRENT_TIMESTAMP( L KI H HAE 
间 格 式 CYYYY-MM-DD HH:MM:SS) 返回 值 ， 示 例如 下 。 


SELECT NOW(}, SYSDATE(}, CURRENT TIMESTAMP {} 5 


Xe ET A AE OB A ZR 


| NOW() | SYSDATE() | CURRENT TIMESTAMP() | 
下 na Tapana ahh tate amvererevauele. alate + 
| 2012-01-10 15:23:52 | 2012-01-10 15:23:52 | 2012-01-10 15:23:52 | 
十 --------------------- 十 --------------------- 十 --------------------- 十 


1 row in set (0.00 sec) 


UNIX_TIMESTAMPO 函 数 以 UNIX 时 间 惟 的 格式 返回 当前 日 期 ， 或 
者 把 一 个 给 定 日 期 转换 为 UNIX 时 间 愉 的 格式 。UNIX 时 间 惟 的 格式 是 以 
从 UNIX 时 间 戳 《〈 即 1970 年 1 月 1 日 午夜 )》 开始 的 秒 数 来 表示 的 ， 示 例如 
ks 


SELECT UNIX TIMESTAMP(}; 


该 查询 运行 的 时 候 ， 产 生 如 下 所 示 的 结 


i Ls See Qed + 
| UNIX TIMESTAMP(} | 
nd end bia td eani + 
| 1326247953 | 
f-e------------------ + 


1 row in set (0.00 sec) 


下 面 的 得 询 获 取 指 定 日 期 的 UNIX 时 间 惟 。 


SELECT UNIX TIMESTAMP( 1973-12-38 ) 3 


XK A AGAR FZR o 


十 ------------------------------ 1 
| UNIX TIMESTAMP ('1973-12-30') | 
十 ------------------------------ 十 
| 126086400 | 
十 ----------------------------- - + 


1 row in set (0.00 sec) 


当 没 有 任何 选项 的 时 候 ，FROM_UNIXTIMEO 函 数 执行 从 一 个 
UNIX 时 间 惟 到 一 个 完整 的 日 期 时 间 格 式 的 转换 ， 示 例如 下 。 


SELECT FROM_UNIXTIME{ '1326247953' } 5; 


我 们 可 以 使 用 DATE_FORMATO0 函 数 的 格式 选项 来 以 更 为 整齐 的 方 
式 显 示 一 个 时 间 惟 ， 示 例如 下 。 


mysql&gt; SELECT FROM UNIXTIME(UNIX TIMESTAMP(), ‘%D %M %Y at %h:%i:%s’); 





ZA WJA RU P EZR o 


站 十 
| FROM UNIXTIME('1326247953') | 
cei Steere Sine AE garantie. a Sees - + 
| 2012-01-10 21:12:33 | 
十 ----------------------------- 十 


1 row in set {0.00 sec) 


可 以 使 用 DATE_FORMATO0 函 数 的 格式 选项 ， 以 更 好 看 的 方式 显示 
一 个 时 间 惟 ， 示 例如 下 。 


SELECT FROM UNIXTIME(UNIX TIMESTAMP(), '%D %M %Y at %hi%i:%s'): 


执行 这 条 得 询 语句 的 时 候 ， 得 到 如 下 的 结 琳 。 


上 十 
| FROM _UNIXTIME(UNIX TIMESTAMP(}, '%D %M %Y at %h:%i:%s') | 
站 二 十 
| 1@th January 2012 at 09:15:38 | 
十 --------------------------------------------------------- 十 


1 row in set {0.00 sec) 


16.12 ”小 结 


在 本 章 中 ， 我 们 学 习 了 SQL 的 基本 知识 ， 从 表 的 创建 到 操作 记录 。 
表 创 建 命 令 需要 3 块 重 要 的 信息 : 表 名 、 字 段 名 和 字段 定义 。 字 段 定 义 
很 重要 ， 因 为 一 个 设计 良好 的 表 有 助 于 加 快 数据 库 的 速度 。MySQL 有 3 
个 不 同 种 类 的 数据 类 型 : 数字 、 日 期 和 时 间 以 及 字符 串 。 


INSERT 命 令 用 来 同一 个 表 添 加 记录 ， 命 令 指定 了 要 填充 的 表 和 
列 ， 并 且 随 后 定义 了 值 。 当 把 值 放 入 到 INSERT 语 句 时 ， 字 符 串 必须 用 
里 引号 或 双 引 号 括 起 来 。SELECT 命 令 用 来 从 特定 的 表 获 取 记 录 。* 字 人 符 
使 得 我 们 能 够 容易 地 选择 表 中 记录 的 所 有 字段 ， 但 是 ， 我 们 也 可 以 指定 
特定 的 列 名 。 如 果 结 果 集 太 长 ， 而 我 们 指定 了 开始 位 置 以 及 要 返回 的 记 
录 的 数目 ， 那 么 LIMIT 子 句 会 提供 一 种 人 简单 的 方法 来 提取 需要 的 结果 。 
要 排序 结果 ， 可 以 使 用 ORDER BY 子 句 来 选择 要 排序 的 列 。 排 序 可 以 对 
整数 、 日 期 和 字符 串 执 行 ， 按 照 升 序 或 降序 排序 ， 默 认 的 顺序 是 升序 。 
WRIA ta xe I, ZR RFA TE EHF AY II EEN 


如 果 没 有 指定 顺序 ， 结 果 将 会 按照 它们 在 表 中 出 现 的 顺序 来 显示 。 

我 们 可 以 使 用 WHERE 子 句 来 测试 条 件 的 有 效 性 ， 从 而 挑选 和 选择 要 返 
回 的 记录 。 比 较 操 作 符 或 馆 辑 操作 符 可 以 在 WHERE 子 句 中 使 用 ， 并 且 
有 时 候 两 种 类 型 都 用 以 组 成 复合 语句 。 在 一 条 语句 中 从 多 个 表 选 择 记 录 
也 是 比较 融 级 的 ， 这 种 类 型 的 语句 叫做 JOIN， 和 需要 提前 思考 和 规划 以 得 
到 正确 的 结果 。JOIN 的 常见 类 型 是 INNER JOIN. LEFT JOIN 和 RIGHT 
JOIN， 尽 管 MySQL 文 持 很 多 种 不 同类 型 的 JOIN 。 我 们 还 学 习 了 ， 在 操 
作 多 个 表 的 时 候 可 以 使 用 子 租 询 而 不 使 用 JOIN。 


UPDATE 和 REPLACE 命 令 用 来 修改 MySQL 表 中 已 有 的 数据 。 
UPDATE 用 于 改变 特定 列 中 的 值 以 及 根据 特定 的 条 件 来 改变 多 条 记录 中 
的 但 。REPLACE 十 INSERT 语 句 的 一 个 变 体 ， 它 删除 反 一 条 其 有 匹配 的 
主键 的 记录 然后 重新 插入 新 记录 。 在 使 用 UPDATE 来 改变 一 个 列 中 的 值 
时 要 小 心 ， 因 为 悉 了 添加 条 件 将 会 导致 表 中 的 所 有 记录 的 给 定 列 都 被 更 
新 。 


DELETE 语 句 很 窗 单 ， 它 从 表 中 删除 记录 ， 这 也 会 有 风险 ， 因 此 ， 
要 确保 只 把 DELETE 权 限 授予 那些 可 以 担负 责任 的 用 户 。 可 以 在 使 用 
DELETE 的 时 候 指 定 条 件 ， 以 便 只 有 当 WHERE 子 句 中 的 一 个 特定 表达 
式 为 true 的 时 候 才 删除 记录 。 另 外 ， 可 以 使 用 一 条 LIMIT 子 句 来 删除 表 
中 记录 的 一 个 较 小 的 集合 。 如 条 我 们 有 一 个 特别 大 的 表 ， 删 除 部 分 比 删 
除 大 表 中 的 每 条 记录 的 资源 密集 性 要 小 。 


我 们 介绍 了 在 字符 串 、 日 期 和 时 间 上 执行 操作 的 MySQL 函 数 。 如 
果 我 们 在 MySQL 有 一 个 字符 串 要 连接 起 来 或 者 想 要 计算 字符 数 ， 可 以 
使 用 CONCATO、CONCAT_WSO 和 LENGTHO 函 数 。 要 填充 字符 串 或 
者 删除 字符 串 中 的 填充 ， 使 用 RPADO、LPADO、TRIMO、LTRIMO 和 
RRIM0 来 得 到 想 要 的 字符 串 。 还 可 以 使 用 LOCATEO、SUBSTRINGO0、 
LEFT() 和 RIGHT() 函 数 在 一 个 字符 串 中 仁 找 为 一 个 字符 串 的 位 置 ， 或 者 
返回 一 个 给 定 字 符 串 的 一 部 分 。 像 LCASEO、UCASEO、REPEATO 和 
REPLACEO 这 样 的 国 数 ， 也 会 返回 最 禄 的 字符 串 的 变 体 。MYySQL 内 建 
的 日 期 和 时 间 函 数 ， 可 以 内 部 地 格式 化 日 期 和 时 间 以 及 执行 日 期 和 时 间 
算术 ， 这 确实 缓解 了 应 用 程序 的 负担 。 用 于 DATE_FORMATO0O 的 格式 化 
选项 ， 提 供 了 一 种 简单 的 方法 来 从 任何 类 型 的 日 期 字段 产生 一 个 自 定 义 
的 显示 字符 串 。DATE_ADDO 和 DATE_SUBO 函 数 以 及 它们 的 众多 可 用 


时 间 间 隅 类 型 ， 可 以 玫 助 我 们 确定 过 去 或 未 来 的 日 期 和 时 间 。 


此 外 ， 像 DAYO、WEEKO、MONTHO 和 YEARO 这 样 的 函数 ， 对 于 
提取 日 期 的 部 分 以 用 于 WHERE 或 ORDER BY 子 句 也 是 很 有 用 的 。 


16.13 Q&A 


: FLAT APT BOR A 4 eB? 哪些 字符 是 受到 限 
制 的 ? 


A: 数据 库 、 表 和 字段 名 的 最 大 长 度 是 64 个 字符 。 任 何 可 以 用 于 目 
录 名 或 文件 名 的 罕 符 ， 也 都 可 以 用 于 数据 库 名 和 表 名 ， 除 了 /和 .。 这 些 
限制 是 有 必要 的 ， 因 为 MySQL 在 你 的 文件 系统 中 创建 了 目录 和 文件 ， 
这 些 与 数据 库 名 和 表 名 对 应 。 除 了 长 度 ， 字 段 名 方面 没有 字符 限制 。 


: 我 可 以 在 一 条 语句 中 使 用 多 个 函数 吗 ? 例如 ， 把 一 个 连 
nine FTAA AS « 


A: 当然 可 以 ， 只 要 别 筷 了 开始 括号 和 结束 括号 。 下 面 这 个 例子 展 
示 了 如 何 把 master name 表 中 的 firstname 和 1lastname 连 接 起 来 并 变 成 大 


SELECT UCASE(CONCAT WS{' ', firstname, lastname}} FROM master_name; 


结果 会 如 下 所 示 。 


| UCASE(CONCAT WS{' ', firstname, lastname}) | 


| JOHN SMITH | 
| JANE SMITH | 
| JIMBO JONES | 
| ANDY SMITH | 
| CHRIS JONES | 
| ANNA BELL | 
| JIMMY CARR | 
| ALBERT SMITH | 
| JOHN DOE | 


9 rows in set (0.00 sec} 


如 果 只 想 把 lastname 变 为 大 写 ， 则 使 用 如 下 话 句 。 


SELECT CONCAT WS(' ', firstname, UCASE(lastname}} FROM master_name; 
结果 会 如 下 所 示 。 


| CONCAT WS{ ', firstname, UCASE(lastname}) | 


John SMITH 
Jane SMITH 


| Jimbo JONES | 
| Andy SMITH | 
| Chris JONES | 
| Anna BELL | 
| Jimmy CARR | 
| Albert SMITH | 
| John DOE | 


9 rows in set (0.00 sec) 


16.14 RAY 


SEEKER 2 BOTT OR FB BO HAS RER eel. BE I AA A 
识 ， 并 且 开 始 把 知识 用 于 实践 。 


练习 题 
1. 整数 56678685 可 以 是 什么 数据 类 型 ? 


2. 如 何 定义 一 个 只 包含 如 下 字符 串 的 信息 : apple. pear. banana 
Allcherry ? 


3. 从 一 个 表 中 选择 前 25 条 记录 的 LIMIT 子 句 是 什么 ? 选择 接 下 来 
的 25 条 记录 的 LIMIT 子 句 呢 ? 


4 如何 使 用 LIKE 来 进行 一 个 字符 串 的 比较 ， 匹 配 名 字 
为 “John” 或 “Joseph”? 


5. 如 何 显 式 地 引用 名 为 table1 的 表 中 的 一 个 名 为 id 的 字段 ? 


6. 编写 一 条 SQL 语句 ， 它 连接 orders 和 items_ordered 这 两 个 表 ， 
个 表 都 有 一 个 order id 主键 。 从 orders 表 中 ， 选 择 如 下 的 字段 order_ name 
Allorder_date. Mitems_ordered#é'7, i4item_description 2X. 


7. 编写 一 条 SQL 查询 来 查找 一 个 子 字 符 串 “grape" 在 一 个 字符 
串 “applepearbananagrape” 中 的 开始 位 置 。 


8. 编写 一 条 但 询 语 句 ， 它 从 字符 串 “applepearbananagrape” 中 选择 
最 后 的 5 个 字符 。 


解答 
1. MEDIUMINT, INT 或 BIGINT。 
2. ENUM (Capple’, 'pear', ‘banana’, 'cherry’) 
ul 
SET (Capple’, 'pear', ‘banana’, 'cherry'). 
3. LIMIT 0, 25 和 LIMIT 25, 25. 
4. LIKE 'Jo%'. 
5. 在 碍 询 中 使 用 tablel.id 而 不 是 id。 
6. SELECT orders.order_name, orders.order_date, 
items_ordered.item_description FROM orders LEFT JOIN 
items ordered ON orders.order_id = items_ordered.id; 
7. SELECT LOCATE(‘grape’, ‘applepearbananagrape’ ); 


8. SELECT RIGHT(“applepearbananagrape”, 5); 


EA wel 


人 花 点 时 间 来 创建 某 些 示 例 表 并 且 练 习 使 用 基本 的 INSERT 和 SELECT 


a 


第 17 章 ”使 用 MySQL 中 的 事务 和 存储 过 程 


ERER, MÄ FI: 


。 事务 的 基础 知识 以 及 如 何在 MySQL 中 使 用 它们 。 
。 存储 过 程 的 基础 知识 以 及 如 何在 MySQEL 中 创建 和 访问 它们 。 


在 前 一 章 中 ， 我 们 学 习 了 SQL 的 基本 知识 以 及 如 何 使 用 MySQL 命 令 
行 界面 来 执行 查询 和 获取 结案 。 仪 仅 这 些 知识 ， 我 们 可 以 成 功 地 完成 在 
本 书 其 他 各 半 中 建立 的 项 目 。 然 而 ， 当 我 们 要 继续 深入 学 习 ， 并 且 考 虑 
构建 适合 在 企业 环境 中 使 用 的 应 用 程序 的 时 候 ， 可 能 需要 更 多 高 级 的 方 
法 ， 以 便 维 护 数据 的 完整 性 并 且 增 强 应 用 程序 和 MySQL 的 通信 。 


尽 过 本 书 其 他 的 章节 不 会 包含 本 章 中 的 元 素 ， 即 本 章 你 持 示 例 尽 可 
能 地 傈 单 ， 以 便 你 的 基础 知识 足够 牢固 ;但 我 们 可 以 很 容易 地 目 行 更 新 
代码 ， 来 包含 在 本 章 所 学 习 的 信息 。 


17.1 {tA~adezss 


数据 库 事务 只 是 必须 按照 如 下 方式 执行 的 一 组 得 询 :， 如 果 其 中 有 一 
个 查询 没有 执行 完 ， 那 么 所 有 的 查询 都 将 失败 。 例 如 ， 假 设 你 有 3 个 
SQL 语 句 为 一 组 的 查询 ， 第 二 个 查询 取决 于 第 一 个 查询 的 结果 ， 而 第 三 
个 查询 取决 于 第 二 个 查询 的 结果 。 如 果 第 二 个 查询 失败 ， 你 需要 有 一 种 
方法 来 取消 第 一 个 僵 询 的 结果 ; 类 似 的 ， 如 末 第 三 个 查询 失败 ， 你 需要 
取消 第 一 个 查询 和 第 二 个 查询 的 结 


通过 在 数据 库 驱 动 的 应 用 程序 中 建立 事务 性 过 程 ， 我 们 可 以 确保 存 
储 在 数据 库 中 的 数据 的 完整 性 。 下 面 的 各 个 小 节 介 绍 了 通过 命令 行 界面 
和 PHP 函 数 来 使 用 事务 的 过 程 。 


提示 : 








当 使 用 InnoDB 存 储 引 擎 的 时 候 ，MySQL 中 事务 性 的 表 才 是 可 用 的 。 从 MySQL 5.5.5 开 


Ly 


台 ，InnoDB 是 表 的 默认 存储 引擎 CS JLhttp://dev.mysql.com/doc/refman/5.5/en/innodb- default- 
se.html 了 解 更 多 信息 ) 。 如 果 你 的 表 是 MyISAM 类 型 的 ， 它 们 将 不 具有 事务 性 。 


17.1.1 事务 中 使 用 的 基本 语法 
当 考 虑 在 MySQL 中 使 用 事务 的 时 候 ， 我 们 还 需要 理解 如 下 的 关键 
术语 。 
e COMMIT 一 一 这 个 命令 出 现在 事务 中 的 一 系列 查询 的 最 后 ， 只 有 
在 所 有 的 查询 成 功 地 执行 之 后 才 执 行 此 命令 。 
。ROLLBACK 一 一 当 事 务 中 的 系列 查询 的 一 个 或 多 个 失败 时 将 使 用 


这 个 命令 ， 并 且 把 相关 的 表 恢 复 到 事务 之 前 的 状态 。 


回顾 一 下 前 面 使 用 的 例子 ， 即 3 个 彼此 依赖 的 查询 ，MySQL 命 令 行 
界面 中 的 一 系列 事件 如 下 所 示 。 


1. 执行 BEGIN 命 令 开始 一 个 新 的 事务 。 
2. 从 tablel 中 选择 一 个 值 以 插入 到 table2 中 。 


3. 如 末 没 有 从 tablel 中 选择 一 个 值 ， 执 行 一 条 ROLLBACK 命 令 ， 
以 确保 事务 结束 并 且 表 返回 到 之 前 的 状态 。 


4. 如 果 从 tablel 中 选择 了 一 个 值 ， 将 这 个 值 插 入 人 到 table2。 


5. 如 果 问 table2 中 插入 一 条 记录 失败 ， 执 行 一 条 ROLLBACK 命 
令 ， 以 确保 事务 结束 并 且 表 返回 到 之 前 的 状态 。 


6. 如 果 一 个 值 插入 到 tablel 中 ， 再 把 这 个 值 插 入 到 table2。 


7. 如 果 把 一 条 记录 插入 到 table3 失 败 ， 执 行 一 条 ROLLBACK 命 
令 ， 以 确保 事务 结束 并 且 表 返回 到 之 前 的 状态 。 


8. 如 果 一 条 记录 成 功 插入 到 table3， 执 行 一 条 COMMIT 命 令 ， 确 保 
事务 结束 并 且 这 个 表 正 确 地 更 新 。 


要 了 解 有 关 MySQL 中 的 事务 的 内 部 工作 机 制 ， 请 参考 位 于 
http://dev.mysql.com/doc/ refman/5.0/en/transactionalcommands.html 的 
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在 下 面 一 下 中 ， 我 们 将 看 到 涉及 到 库存 和 销售 记录 的 表 使 用 事务 的 
例子 。 


17.1.2 ”使 用 事务 的 例子 


假设 你 已 经 创建 了 一 个 在 线 的 商店 ， 它 拥有 保存 了 库存 、 销 售 记录 
和 和 销售 记录 的 细 目 的 数据 库 表 。CREATE 语 句 如 下 所 示 。 


CREATE TABLE store_inventory { 
id int not null primary key auto_increment, 
item name varchar(5@), 
item price float(6,2), 
item_qty int 
) ENGINE=InnoDB; 


CREATE TABLE store_orders ( 
id int not null primary key auto_increment, 
purchaser_name varchar(5@}, 
purchase date datetime 

) ENGINE=InnoDB; 


CREATE TABLE store orders items { 
id int not null primary key auto_increment, 
order_id int, 
inventory id int, 
item qty int 
) ENGINE=InnoDB; 


在 这 个 例子 的 store_inventory 表 中 ， 我 们 可 以 找到 如 下 的 两 条 记 


Ko 

MANNS Maon EEE AERAN ers, ea ste + 
| id | item_name | item_price | item_qty | 
te 4 ti + 
| 1 | Great Book | 19.99 | 10 | 
| 2 | Awesome CD | 9.99 | 20 | 
人 和 Pina inne ee 1 


Yr RS i SE OR EY A E Je WW SE PA Great Books 和 一 个 
Awesome CD， 过 程 如 下 上 所 示 。 


1. 用 户 完 成 一 个 在 线 表单 并 且 试 图 为 购买 而 进行 文 付 ， 因 此 ， 执 
行 一 条 BEGIN 命 令 以 使 一 个 事务 成 为 结账 脚本 的 一 部 分 ， 如 下 上 所 示 。 


BEGIN; 
2. 4Estore_inventory#< "F HY fal m HY A Be a EFA DV H o 


UPDATE store inventory SET item qty = item_qty - 2 WHERE id 
UPDATE store inventory SET item qty = item_qty - 1 WHERE id 


Hl 
一 人 
wa 


3. [HJstore_orders# Zs — Kid: 


INSERT INTO store _ orders (purchaser_name, purchase date) 
VALUES (‘John Smith’, now(}}; 


4. 如果 添加 记录 失败 ， 执 行 一 条 ROLLBACK 命 令 来 重 置 可 供 交 易 
的 商品 的 数量 ， 如 下 所 示 。 


ROLLBACK; 


5. WRI, RAWSON cS AID, FF Ate 
store_orders_lineitems 表 中 插入 记录 ， 从 而 在 同 销售 记录 添加 细 目 的 查询 
中 用 到 这 个 ID， 如 下 所 示 。 


INSERT INTO store orders items (order id, inventory 1id, item qty) 
VALUES (1，1，2 ); 

INSERT INTO store orders items (order id, inventory id, item qty) 
VALUES (‘1', ‘2°, 1 ); 


6. 如 果 添 加 记录 失败 ， 执 行 一 条 ROLLBACK 命 令 来 恢复 商品 可 供 
交易 的 数量 并 且 把 store_orders 中 的 记录 删除 ， 如 下 所 示 。 
ROLLBACK: 


7. 如 果 添 加 记录 成 功 ， 但 是 后 续 的 信用 卡 文 付 或 者 其 他 文 付 方法 
失败 ， 执 行 一 条 ROLLBACK 命 令 来 恢复 商品 可 供 交 易 的 数量 ， 把 


store_orders 中 的 记录 删除 ， 并 且 把 store_orders_lineitems 中 的 记录 删除 ， 
如 下 所 示 。 


ROLLBACK; 


8. 如 果 成 功 地 添加 了 一 条 记录 ， 并 且 后 续 的 信用 卡 支 付 或 其 他 的 
支付 方式 也 成 功 了 ， 执 行 一 条 COMMIT 命 令 来 确保 所 有 的 改变 都 存储 并 
用 事务 结束 ， 如 下 上 所 示 。 


COMMIT ; 


当然 ， 一 个 在 线 商店 不 会 通过 命令 行 界 面 百 接 和 MySQL 交 互 ， 但 
是 ， 会 通过 像 PHP 这 样 的 脚本 语言 来 交互 。 但 是 ， 如 过 你 理解 了 事务 背 
后 的 过 程 ， 将 PHP 加 入 其 中 很 简单 ， 只 需要 执行 前 面 列 出 的 查询 和 命 
令 。 任 何其 他 的 PHP 和 MySQL 的 交互 也 没有 什么 不 同 ， 我 们 会 在 第 18 章 
中 学 习 到 这 些 。 


除了 我 们 将 在 第 18 章 中 竺 习 的 内 容 ， 如 末 想 要 在 目 己 的 脚本 中 使 用 
事务 ， 请 确保 浏 多 一 下 PHP 手 册 中 的 如 下 这 些 函 数 的 定义 。 


e mysqli_autocommit() —http://www.php.net/mysqli_autocommit 
e mysqli commit() —http://www.php.net/mysqli_commit 


e mysqli_rollback() —http://www.php.net/mysqli_rollback 


还 是 要 强调 一 下 ， 一 定 要 看 一 下 MySQL 手 册 以 了 解 关 于 事务 的 更 
多 信息 ， 参 见 http://dev.mysql.com/doc/refman/5.5/en/sql-syntax- 
transactions.html ， 尤 其 是 讨论 不 能 回 深 〈 管 理性 操作 〉 的 事务 类 型 ， 知 
着 其 中 大 多 数 内 容 症 有 用 的 。 


17.2 什么 是 存储 过 程 


简 蛙 地 说 ， 和 存储 过 程 是 存储 在 数据 库 服 务 占 上 而 个 是 Web 服 务 桌 上 
的 一 个 SQL 编 写 的 过 程 。 你 可 能 会 认为 ， 不 会 在 Web 服 务 间 上 和 存储 任何 
过 程 ， 但 是 实际 上 ， 任 何 包 含 SQL 查询 的 脚本 都 作为 一 个 过 程 存储 在 
Web 服 务 磺 上 。 例 如， 应 用 程序 中 选择 、 删 除 、 更 新 或 插入 数据 到 表 中 
的 每 个 得 询 《〈 我 们 半 立 各 天 输入 到 脚本 中 的 代码 ) ， 将 会 作为 存储 过 程 
存储 到 数据 奋 中 ， 并 且 在 脚本 中 5 引用。 


在 代 人 码 中 使 用 存储 过 程 的 支 持 者 指出 ， 性 能 和 可 维护 性 是 这 么 做 的 
关键 原因 。 





。 更 好 的 性 能 存储 过 程 作为 一 个 预 编译 的 SQL 存 在 于 数据 库 
中 ， 因 此 ， 一 个 典型 的 两 步 过 程 〈“ 编 译 和 执行 ) 变 成 了 单 步 过 程 
(执行 )。 

。 另 于 维护 一 一 在 一 个 地 方 ( 数 据 库 中 ) 维护 一 条 语句 ， 比 在 多 个 
地 方 ( 如 Web 服 务 占 的 各 个 脚本 中 ， 维护 一 条 语句 要 少 伦 很 多 时 
间 。 此 外 ， 把 所 有 这 些 语句 存储 到 数据 库 中 ， 而 不 是 存储 到 Web 服 
务 规 文档 根 目 录 下 的 实际 的 文本 文件 中 ， 这 融 多 了 一 条 你 护 指 施 ， 
以 防止 有 人 访问 你 的 web 服务 左上 的 文件 ， 这 样 一 来 ， 他 们 上 所 拥有 
的 只 是 调用 存储 过 程 的 俘 询 ， 而 个 古 过 程 本 里 的 逻辑 。 


一 个 有 用 的 存储 过 程 的 例子 ， 束 古 用 来 生成 条 种 类 型 (可 能 十 财 务 
数据 、 销 售 库 存 或 者 其 他 的 类 型 的 一 个 报表 的 SQL 合 询 ， 想 象 一 下 那 
将 是 包含 了 很 多 过 程 的 一 个 复 洒 合 询 。 创 建 一 个 存储 过 程 ， 而 不 是 这 种 


FRAN — TSE, aS BI TEPER eh. UR TG “ME 
“SDF ee PARKEARE, AE BSE TE, AR 
BI FF fi OE EN AEP VERN Ah MET re ee tid FN be SOR, El 
EAE H E ABH E SF EEEE HILFE o 


如 下 定义 的 表 将 用 于 这 个 存储 过 程 的 例子 中 。 


CREATE TABLE test&P 1 
id int not null primary key auto_increment, 
Field name varchar(25), 
date added datetime 

) ENGINE=InnoDB; 


这 张 用 于 测试 的 表 中 的 记录 如 下 。 


ee EO E E E te ee E hs ey 人 + 
| id | field name | date added | 
+----t------------ 十 --------------------- 十 
| 1 | Value 1 | 2012-01-23 09:40:24 | 
| 2 | Value 2 | 2012-01-24 09:40:24 | 
| 3 | Value 3 | 2012-01-25 09:40:24 | 
| 4 | Value 4 | 2012-01-26 09:40:24 | 
| 5 | Value 5 | 2012-01-27 09:40:24 | 
| 6 | Value 6 | 2012-01-30 09:40:24 | 
| 7 | Value 7 | 2012-01-31 09:40:24 | 
| 8 | Value 8 | 2012-02-01 09:40:24 | 
| 9 | Value 9 | 2012-02-02 09:40:24 | 
| 10 | Value 10 | 2012-02-14 09:40:24 | 
BR city ch en E nel as ts E ee + 


要 将 这 个 示例 表 转 换 为 存储 过 程 ， 接 下 来 ， 必 须 确保 MySQL 知 道 
我 们 将 要 在 存储 过 程 中 使 用 的 分 隔 符 。 这 个 例子 中 使 用 %/" 作 为 分 陋 
人 符 ， 因 此 ， 必 须 执 行 如 下 的 得 询 。 


DELIMITER // 


创建 一 个 基本 的 存储 过 程 的 语法 如 下 。 


CREATE PROCEDURE procedure name () query // 


对 于 这 个 例子 ， 这 个 存储 过 程 只 是 从 testSP 表 选取 所 有 的 数据 ， 我 
们 把 这 个 存储 过 程 命 名 为 9p1， 定 义 如 下 。 


CREATE PROCEDURE sp1 () SELECT * FROM testSP WHERE date added BETWEEN 
DATE SUB(NOW{), INTERVAL 7 DAY) AND NOW() // 


要 调用 这 个 存储 过 程 ， 使 用 CALEL 人 命令， 示例 如 下 。 


CALL Sp1 () ji 


这 个 存储 过 程 《SELECT 查询) 的 结果 会 返回 给 你 ， 如 下 所 示 。 
和 OE + 
| id | field name | date added | 
bes ob ee i ay a Posse ee eee ood eae + 
| 2 | Value 2 | 2012-01-24 09:40:24 | 
| 3 | Value 3 | 2012-01-25 09:40:24 | 
| 4 | Value 4 | 2012-01-26 09:40:24 | 
| 5 | Value 5 | 2012-01-27 09:40:24 | 
| 6 | Value 6 | 2012-01-30 09:40:24 | 
Be ak ey carry a et As, lS, gS: tes ay rier ao, Skt ah ee N + 


在 第 18 章 中 ， 我 们 将 学 习 使 用 PHP 执 行 这 些 SQL 查 询 的 过 程 。 显 
然 ， 这 短 短 的 几 页 对 于 使 用 存储 过 程 来 说 甚至 算 不 上 隔 肤 挠 痒 ， 本 贡 
介绍 概念 。 其 他 的 内 容 推荐 阅读 MySQL 手册 中 关于 存储 过 程 的 P 


Z, 4\i-Fhttp://dev.mysql.com/doc/refman/5.5/en/stored- routines.html 。 


EA 
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这 个 简短 的 一 章 简 单 地 介绍 了 在 使 用 mnoDB 人 存储 引擎 的 时 候 ， 
MySQL 中 事务 处 理 的 概念 以 及 存储 过 程 的 使 用 。 除 了 简单 地 介绍 了 这 
两 个 概念 ， 本 章 还 使 用 了 真实 的 例子 来 搞 述 这 两 个 局 级 话题 。 本 重 的 全 
部 目标 是 : 介绍 一 些 在 你 目 己 开 肥 企业 级 应 用 的 时 候 应 该 理解 的 一 些 概 
念 和 术语 。 在 练习 本 书 中 的 示例 的 时 候 ， 记 住 这 些 知识 ， 并 且 笑 试 找到 
方法 来 改进 以 前 的 基础 查询 。 


17.4 Q&A 


Q: 既然 MySQL 支 持 事务 ， 我 必须 随时 使 用 事务 吗 ? 


A: 不 是 的 ， 特 别 是 如 来 你 的 应 用 程序 和 站 扣 的 动态 方面 是 为 了 动 
态 显 示 数 据 ， 而 不 是 为 了 动态 插入 数据 的 话 ， 不 一 定 要 使 用 事务 。 此 
外 ， 如 条 数据 的 插入 不 一 定 是 和 财务 或 库存 相关 的 业务 ， 也 不 必 使 用事 
务 。 换 句 话 说， 如 条 你 没有 使 用 事务 并 且 一 次 插入 或 更 新 租 询 失败 的 
话 ， 你 要 确保 这 个 失败 无 伤 大 雅 ， 不 会 给 你 市 来 经 鹿 损 失 ， 也 不 会 导致 
你 失去 重要 客户 的 数据 。 


17.5 ”实践 练习 


实践 练习 是 设计 用 来 帮助 你 预料 可 能 的 问题 、 复 习 已 经学 过 的 知 
识 ， 并 且 开 始 把 知识 用 于 实践 。 


练习 题 


1. 判断 对 错 : MyISAM 是 MySQL 中 默认 的 、 完 全 事务 性 的 存储 引 


党， 


2. 如 果 包 含 3 步 操 作 的 事务 中 的 第 2 步 失 败 了 ， 应 该 执行 哪 条 命 
NF 
Ye 


3. SEA PALE PATA 2 ITS ? 


万 Ar 
解 管 
1. 错 。ImnoDB 是 默认 的 存储 引擎 并 文 持 完全 事务 。 
2. ROLLBACK 


3. 性 能 更 好 和 多 于 维护 。 


EA wel 


1. 想 想 可 能 需要 事务 的 其 他 一 些 数 据 库 交 互 类 型 ， 并 且 ， 像 本 章 
中 一 样 ， 创 建 你 目 己 的 表 集 合 。 壬 试 在 你 可 能 第 要 调用 和 回 深 命 令 的 所 
有 秘 辑 位 置 使 用 它们 。 


2. 创建 一 个 和 存储 过 程 〈 太 其 所 需 的 表 〉 ， 它 报告 在 两 个 给 定 日 期 
之 间 所 友 生 的 一 个 特定 贡 物 的 所 有 销售 事务 。 


第 18 章 ”使 用 PHP 和 MySQL 交 互 


ERER, MÄ FI: 


。 如 何 使 用 PHP 连 接 到 MySQL。 
。 如 何 通 过 PHP 脚 本 插入 和 查询 数据 。 


既然 我 们 已 经 学 习 了 PHP 的 基础 知识 和 使 用 MySQL 的 基础 知识 ， 这 
就 为 学 习 二 者 之 间 的 交互 做 好 了 准备 。 把 PHP 看 作 是 通 回 MySQL 的 一 个 
管道 ， 我 们 在 上 一 和 章 学 习 到 的 命令 耽 是 将 要 在 本 和 章 及 大 到 MySQL 的 命 
令 ， 只 是 这 次 我 们 使 用 PHP 有 友 送 它们 。 


18.1 MySQL er 20 4iIMySQLirk 2% 


如 果 你 使 用 过 PHP 的 旧版 本 ， 或 者 使 用 的 是 PHP 的 当前 版 本 和 
MySQL 的 旧版 本 ， 可 能 会 熟悉 mysql_* 函 数组 。 你 可 能 还 需要 在 互联 网 
上 找到 一 些 使 用 mysql_* 函 数组 的 示例 代码 。 


然而 ， 从 MySQL 4.1.3 版 开始 (现在 已 经 过 去 7 年 7 了) ，MYySQL 数 
据 库 系 统 包 含 了 在 PHP 中 强制 使 用 新 通信 方法 的 功能 ， 这 些 方法 全 部 包 
含 在 mysqli_* 孙 数组 中 。 


本 章 中 的 所 有 代码 ， 以 及 本 书 其 他 部 分 的 代码 ， 痢 使 用 mysqli_* 也 | 
数组 。 要 了 解 评 细 信息 ， 请 参阅 位 于 http:/www.php.net/mysqli 的 PHP 手 
WHR EA 


18.2 ”使 用 PHP 连 接 MySQL 


要 成 功 地 使 用 PHP 函 数 和 和 MySQL 交互， 必须 在 Web 服 务 嚣 能够 连接 
到 的 一 个 位 置 〈 不 一 定 必 须 和 Web 服 务 器 位 于 相同 的 机 器 ) 运行 
MySQL。 还 必须 创建 一 个 MySQL 用 户 《〈 这 有 一 个 密码 ) ， 并 且 必 须知 
道 想 要 连接 的 数据 库 的 名 字 。 如 果 间 从 第 2 章 和 第 4 章 的 说 明 ， 我 们 应 该 
已 经 做 好 了 这 些 准备 。 如 果 PHP 和 MySQL 托 管 于 一 个 Internet 服 务 器 提 
供 商 ， 请 在 处 理 之 前 ， 通 过 系统 管理 员 确保 你 使 用 了 正确 的 用 户 名 、 密 
AGU) RE FEA o 


在 本 草 的 所 有 示例 脚本 中 ， 示 例 数据 库 的 名 字 是 testDB， 示 例 用 户 
是 joeuser， 而 示例 密码 是 somepass。 使 用 这 些 脚 本 的 时 候 ， 请 用 你 目 己 
的 信息 符 换 它们 。 


提示 : 








本 章 中 的 所 有 代码 以 及 其 他 章 市 的 代码 都 进一步 反映 了 过 程式 的 mysqgli_* 孔 数组 的 使 用 。 
也 可 以 以 面 问 对 象 的 方式 来 使 用 这 些 国 数 ， 要 了 解 这 方面 的 更 多 信息 ， 请 阅读 位 于 
http://www.php.net/mysqli 的 PHP 手 册 。 如 果 你 是 从 一 种 面 回 对 象 编程 语言 或 者 有 和 面 癌 对 象 思想 
的 编程 语言 转 同 PHP 的 ， 我 建议 你 阅读 一 下 PHP 手 册 中 的 面 回 对 象 功能 介绍 ， 并 且 在 适当 的 时 
候 用 它 来 蔡 换 ， 从 概念 上 讲 ， 这 些 过 程 都 是 相似 的 。 








然而 ， 如 果 你 是 编程 新 手 ， 或 者 还 没有 接触 面 回 对 象 编程 忠 想 ， 在 你 的 日 党 工作 中 学 习 
并 使 用 过 程式 方式 古 没 问题 的 。 在 本 书 中 ， 我 继续 使 用 过 程式 编程 。 因 为 ， 对 于 程序 员 新 手 
理解 过 程 来 说 ， 这 已 经 证 明 是 最 好 的 方法 。 





18.2.1 ”进行 连接 


连接 到 MySQL 的 基本 语法 如 下 。 


$mysqli = mysqli connect{"hostname", "username", "password", “database"}; 


$mysqli 的 值 是 函数 的 结果 ， 并 且 随 后 用 在 与 MySQL 通 信 的 函数 
H, 


以 一 个 实际 的 示例 值 ， 连 接 代 人 码 如 下 所 示 。 


$mysqli = mysqli connect{"localhost", "joeuser", "somepass", "testDB"}; 


程序 清单 18.1 是 连接 脚本 的 一 个 有 效 例子 。 它 在 第 2 行 创建 了 一 个 
狐 的 连接 ， 然 后 ， 测 斌 是否 发 生 了 一 个 错误 。 如 果 友 生 了 错误 ， 第 5 行 
显示 出 一 条 错误 消息 ， 并 有 使 用 mysqli_connect_errorO 函 数 来 录 示 消 
思 。 如 采 没 有 及 生 错 误 ， 第 8 行 显 示 一 条 消息 ， 其 中 包含 了 调用 

mysqli_get_host_info() i274 2) ENUE E o 


程序 清单 18.1 一 个 简单 的 连接 脚本 


1: <?php 

2: $mysqli = new mysqli{ "localhost", "joeuser", "“somepass", "testDB"); 
a 

4: if (mysqli_connect_errno(}) { 

S printf("Connect failed: %s\n", mysqli_connect_error()); 

6: So 

7: } else { 

2 printf("Host information: %s\n", mysqli get host _info($mysqli)}; 
9: } 

10: ?> 


把 这 个 脚本 保存 为 mysqlconnect.php， 并 且 将 其 放置 到 Web 服 务 器 的 
文档 区 域 。 使 用 Web 浏 虎 震 访问 这 个 脚本 ， 如 果 连 接 成 功 的 话 ， 将 会 看 
到 如 下 所 示 的 结果 。 


Host information: localhost via TCP/IP 


你 可 能 还 会 看 到 如 下 内 容 。 
Host information: Localhost via UNIX socket 
QRIERAM, Keita ARIAS. OT 
mysqli_connect_error()Pi aur" “Matix, Bil Pu Bata. 


Connect failed: Access denied for user ‘joeuser'@'localhost' (using password: 
YES) 


然而 ， 如 条 连接 成 功 ， 第 8 行将 显示 mysdli_get_host_info0 的 输出 ， 
WR ETAL EY 9 FE o 

FR ASST se ES eR - MEO RS, (Ae, WSOC 
连接 是 个 好 主 是 。 我 们 可 以 在 程序 清单 18.2 的 第 9 行 看 到 如 何 做 到 这 一 


/NO 


程序 清单 18.2 ”修改 后 的 简单 连接 脚本 


1: <?php 

2: $mysqli = new mysqli{"localhost", "joeuser", "Somepass", "testDB"}); 
a: 

4: if (mysqli_connect_errno(}) { 

D: printf("Connect failed: %s\n", mysqli connect_error()); 

6: exit); 

7: + else 1 

8: printf ("Host information: %s\n", mysqli _get_host_info($mysqli)}; 
9: mysqli close($mysqli) ; 

10: } 

IE: ae 


但 是 在 第 5 行 之 后 ， 我 们 没有 使 用 mysgl_close(0) 函 数 ， 这 是 因为 如 时 
执行 了 第 5 行 ， 当 初 束 没有 建立 连 返 。 对 于 使 用 PHP 连 接 MySQL 的 介绍 
Pe Bae 了 。 下 一 节 介 绍 合 询 执行 函数 ， 这 比 人 简单 地 打开 一 个 连接 什 
么 也 不 做 要 有 趣 得 多 。 


18.2.2 Tew 


知道 如 何 编写 SQL 并 且 学 习 了 本 章 前 面 的 基本 知识 ， 那 么 使 用 PHP 
执行 MySQL 查 询 的 任务 就 进行 一 半 了 。PHP 中 的 函数 mysqli_query0) 用 
KI] MySQLAIFSQLEA WH .- 


在 脚本 中 ， 首 先进 行 连接 ， 然 后 执行 一 个 得 询 。 程 序 清 单 18.3 中 的 
脚本 创建 了 一 个 多 为 testTable 的 示例 表 。 


程序 清单 18.3 ”创建 一 个 表 的 脚本 


1: <?php 

2: mysqli = mysqli _connect("localhost", "“joeuser", "somepass", "testDB"); 
3: 

4: if (mysqli connect errno(}} { 

5 printf ("Connect failed: %s\n", mysqli _connect_error()); 
G: exit(); 

7: } else { 

8: $sql = "CREATE TABLE testTable 

9; (id INT NOT NULL PRIMARY KEY AUTO INCREMENT, 

10 : testField VARCHAR({75))"; 

i $res = mysqli query($mysqli, $sql); 

I2: 

13: if ($res === TRUE) { 

14: echo "Table testTable successfully created."; 
a: } else { 

16: printf("Could not create table: %s\n", mysqli _error($mysqli) ) ; 
1 ee } 

ee 

19: mysqli _close($mysqli); 

20: } 

ely T> 








通过 一 个 脚本 执行 查询 的 时 候 ，SQL 语 句 末尾 的 分 号 不 是 必须 的 。 


在 第 8 行 到 第 10 行 组 成 SQL 语句 的 文本 并 赋 给 $sql 变 量 。 这 种 做 法 坪 


随意 的 ， 你 长 至 不 希 要 把 SQL 碍 询 的 内 容 放 到 一 个 单独 的 变量 中 ， 这 个 
例子 中 这 么 做 只 是 为 了 使 这 一 过 程 的 各 个 步 又 更 清楚 。 


mysqli_gquery 肖 数 返回 一 个 值 ，true 或 false， 从 第 13 行 开始 的 if...else 
语句 检查 这 个 值 。 如 果 $res 的 值 为 tue， 束 会 在 屏 贤 上 显示 一 条 成 功 的 
消息 。 如 果 我 们 通过 命令 行 界 面 访问 MySQL， 以 验证 testTable 表 的 创 
建 ， 将 会 看 到 DESCRIBE test Table 命 令 的 如 下 输出 。 


re en eee sions lage ees. emi gerbe Shere tienes. jeere ts eeens ahaa Pete ihre, Shuai amen sR: + 
| Field | Type | Null | Key | Default | Extra 
A 人 +------ ss he i ts an cick + 
| id | int(11) | | PRI | NULL | auto increment | 
| testField | varchar(75}) | YES | | NULL | 
se ee ee 人 人 ere ee eee Saree Srna eae EE + 


如 果 是 这 种 情况 ， 恭 喜 你 ， 你 已 经 使 用 PHP 在 MySQL 数 据 库 中 成 功 
地 创建 了 一 个 表 。 


然而 ， 如 果 $res 的 值 不 为 tue， 将 会 显示 一 条 钳 误 的 消息 ， 这 条 闹 
已 由 mysqli_errorO 函 数 产 生 。 
18.2.3 ”获取 错误 消息 


化 点 时 间 熟 悉 一 下 mysgli_error() 水 数 ， 它 将 会 成 为 你 的 朋友 。 当 和 
PHP die() 函 数 〈 在 过 到 该 函数 的 时 候 和 直接 退出 脚本 ) 一 起 使 用 的 时 候 ， 
如 末 出 现 错误 ，mysqli_errorO 函 数 将 会 返回 一 条 有 用 的 错误 信息 。 


例如 ， 既 然 我 们 已 经 创建 了 一 个 名 为 testTable 的 表 ， 束 可 以 再 次 执 
行 该 脚本 而 没有 任何 错误 。 演 试 再 次 执行 该 脚本 ， 应 该 会 在 Web 浏 贤 妖 
中 看 到 类 似 下 和 面 的 结果 : 


Could not create table: Table testtable already exists 


BARMAN! REST ROW, FPR TA Bo, BVI 
eI PHP SRM a IPR CIE « 


18.3 ”使 用 MySQL 数 据 


tH A. ar. AHR ASR A ACHE BB Fl SE (2 A mysqli_query() PAI BOK FA 
行 我 们 在 第 16 章 学 习 过 的 基本 的 SQL 查询 。 对 于 INSERT、UPDATE 和 
DELETE 奉 询 ， 在 得 询 执行 后 不 需要 额外 的 脚本 ， 因 为 我 们 不 要 显示 任 
何 结果 (除非 想 要 这 么 做 ) 。 使 用 SELECT 查询 的 时 候 ， 有 几 个 选项 用 
来 显示 得 询 所 获取 的 数据 。 让 我 们 从 基本 的 插入 数据 开始 ， 这 样 下 有 了 
可 供 访 问 的 内 容 。 


18.3.1 ERSQLIEA 


在 程序 清单 18.3 的 创建 表 脚 本 中 ，SQL 查 询 中 使 用 的 数据 是 直接 编 
码 到 脚本 中 的 。 然 而 ， 你 可 能 要 构建 的 是 动态 Web 站 点 或 者 基于 Web 的 
应 用 程序 ， 这 时 候 ， 你 很 可 能 是 根据 一 个 表单 的 用 户 输 入 或 者 其 他 的 过 
程 ， 向 表 中 插入 数据 或 者 从 表 中 查询 数据 。 如 果 你 没有 留意 用 户 输入 ， 
并 且 在 查询 中 使 用 用 户 输入 之 前 没有 进行 安全 性 检查 ， 那 么 ， 你 可 能 会 
遭受 SQL 注入 式 攻 击 。 


当 个 列 怀 有 恶意 的 人 信 机 在 你 的 表单 字段 中 输入 整个 或 部 分 SQL 碍 
询 的 时 候 ， 融 会 及 生 SQL 注 入 云 攻 击 ;， 我 们 假设 执行 这 些 碍 询 的 时 候 ， 
安全 性 会 受到 破坏 ， 数 据 有 烘 串 的 次 在 人 危险。 


一 篇 知名 的 XKCD 网 络 漫画 《Little Bobby Tables》， 很 好 地 说 明了 
SQL 注入 式 攻 击 的 问题 。 各 大 讨论 论坛 和 其 他 编程 相关 的 帮助 站 点 经 第 
引用 这 段 漫画 ， 并 且 在 针对 表单 输入 和 查询 相 关 的 问题 给 出 解答 的 时 
候 ， 都 配 上 相应 的 说 明 ,，“Don't forget Little Bobby Tables!”. FJ LA 


fEhttp://xkcd.com/327/  #1xX Ai H] - 


看 下 面 的 示例 ， 它 试图 从 一 个 叫做 users 的 表 收 集 用 户 信息 ， 其 中 
name 字 段 与 表单 中 输入 的 一 个 值 匹配 ; 这 很 像 是 一 个 Web 登 录 过 程 。 


SELECT * FROM users 
WHERE name = '".$ POST[ ‘username from_form']."'; 


假设 username from form 字 段 中 的 值 如 下 所 示 。 
or '1'='1 
这 会 产生 如 下 的 一 个 完整 查询 。 


SELECT * FROM users 
WHERE name = = = or {qs 3 


Xa rear eT SU, VAL Le AR 


你 可 能 已 经 明日 了 ， 四 eee PHP 手册 中 关于 SQL 注 入 式 
攻击 的 页 none 不 有 更 多 的 示例 ， 
本 php 。 在 整个 
本 书 中 ， 代 码 示 例 都 限制 了 了 SQL 注入 式 攻 击 的 可 能 性 ， 只 有 一 个 例外 ， 
那 殉 是 显示 错误 消息 。 随 独 你 不 断 和 学 习 ， 并 且 在 一 个 开 及 环境 而 不 是 产 
同 坏 境 中 操作 ， 我 建议 你 将 错误 消 恩 打印 到 屏 舌 上 ， 以 便 理 解 发 生 了 什 
ZA (MIRA REA) 。 在 产品 环境 中 ， 你 应 该 隐 疾 错误 消息 ， 特 别 
fe SF RA I a as a E H Be A SAR 


“VK Je S URETA EE SUT SA A MySQL #UPHP i ets 
后 ， 看 一 下 PDO(PHP Data Objects) 抽 象 层 ， 以 进一步 巩固 你 的 产品 应 用 
程序 ， 参 见 http:/ www. php. net/ manual /en/book.pdo.php 。 一 个 好 的 开 
始 的 地 方 ， 可 能 是 天 于 预定 义 语句 和 存储 过 程 的 部 分 ， 参 见 http:// 


www.php.net/manual/en/pdo.prepared-statements.php 。 


18.3.2 ”使 用 PHP 插 入 数据 


在 这 个 阶段 ， 插 入 数据 的 最 简单 的 方法 束 是 直接 编写 INSERT 语 
句 ， 如 程序 清单 18.4 所 示 。 





程序 清单 18.4 插入 一 条 记录 的 脚本 


1: <?php 

2: mysqli = mysqli _connect("localhost", "joeuser", "somepass", "testDB"); 
g. 

4: if (mysqli connect_errno(}) { 

5: printf("Connect failed: %s\n", mysqli_connect_error()); 

6: exit(); 

7: } else { 

8 : $sql = "INSERT INTO testTable (testField) VALUES ('some value')"; 
9: $res = mysqli query($mysqli, $sql); 

10: 

1 if ($res === TRUE) { 

le: echo "A record has been inserted." ; 

t33 } else { 

14: printf("Could not insert record: %s\n", mysqli _error($mysqli) ) ; 
tas } 

16: 

if: mysqli close($mysqli); 

tox 3 

igs Ye 


AIFS PTR 18.340 EE, eS AAS E48 ie HR A ioe, ff 
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存储 到 $sql 变 量 中 ， 而 文本 都 是 在 第 12 行 和 第 14 行 修改 。 连 接 代 人 码 和 的 
行 得 询 的 结构 都 是 相同 的 ， 实 际 上 ， 连 接 MySQL 的 大 多 数 过 程式 代码 
祁 将 会 于 从 相同 类 型 的 代码 模板 。 


把 这 个 脚本 命名 为 mysqlinsert.php， 然 后 将 其 放 入 到 Web 服 务 器 
中 。 运 行 这 个 脚本 将 会 使 testTable 表 增加 额外 的 一 行 。 要 输入 比 脚本 中 


Haws Rida, FSi SER Beam SSQL ia WN AIF A AR 
( Fmysqli_queryQ PA BOR PUT Hea A), Bae AT Ail ac iis DR AS 
一 个 基于 表单 的 界面 。 


要 为 这 个 脚本 创建 表单 ， 硝 实 只 需要 一 个 字段 ， 因 为 id 字段 可 以 目 
动 增加 。 表 单 的 动作 了 驶 是 记录 添加 脚本 的 名 字 ， 我 们 把 它 叫 做 
insert.php。HTML 表 单 看 上 去 如 程序 清单 18.5 所 示 。 


程序 清单 18.5 一 个 插入 表单 


<form action="insert.php" method="POST"> 

<p><label for="testfield">Text to Add:</label><br/> 

9: <input type="text" id="testfield" name='"testfield" size="30" /></p> 

10: <button type="submit" name="sSubmit" value="insert">Insert Record</button> 
11: </form> 

12: </body> 

13: </html> 


1: <!DOCTYPE html> 

2: <html> 

3: <head> 

4: <title>Record Insertion Form</title> 
5: </head> 

6: <body> 

Ta 

Oo 


把 这 个 文件 保存 为 insert_form.html， 并 且 将 其 放置 到 Web 服 务 器 的 
文档 根 目 孙 下 。 接 下 来 ， 创 建 程 序 清单 18.6 所 示 的 insert.php 脚 本 。 表 单 
中 输入 的 值 将 会 通过 一 个 名 为 $ POST [testfield] 的 变量 来 替换 SQL 查询 
中 直接 编码 的 值 。 


程序 清单 18.6 ”和 表单 一 起 使 用 的 一 个 插入 脚本 


ON Oo Bw DY 


<?php 
$mysqli = mysqli connect("localhost", "joeuser", "somepass", "testDB"); 


if (mysqli_connect_errno(}) { 


printf("Connect failed: %s\n", mysqli _connect_error()); 
exit(); 


} else { 


$clean text = mysqli real escape string({$mysoqli, $ POST[ 'testfield']}); 


$sql = "INSERT INTO testTable (testField) 
VALUES ('".$clean text."')}"; 


$res = mysqli _query($mysqli, $sq1); 


if ($res === TRUE) { 
echo "A record has been inserted.'; 
} else { 


printf("Could not insert record: %s\n", mysqli_error($mysqli) ) ; 


} 


mysqli _close($mysqli) ; 


XNA E 18.477 ASI AS ZT) ESE IIE BIT, H 
中 ， 表 单 的 输入 进行 加 强 以 避免 SQL 注 入 ， 并 且 在 第 10 行 ， 我 们 使 用 了 
安全 有 的 $clean_text 字 从 串 来 蔡 换 前 面 示 例 中 所 使 用 的 直接 编码 的 文本 字 
从 串 。 为 了 使 得 输入 更 安全 ， 我 们 使 用 S mysqli_real_escape_string() 
数 ， 这 个 函数 需要 已 经 建 并 了 连接 ， 因 此 将 它 放 到 了 if...else 语 句 的 else 


部 分 中 。 
把 这 个 脚本 保存 为 inserLphp， 并 且 将 其 放置 在 Web 服 务 器 的 文档 根 


日 录 下 。 在 浏览 器 中 访问 所 创建 的 这 个 HTML 表 单 。 它 看 上 去 如 图 18-1 
所 示 。 





fa Record Insertion Form 


€ > C |© http://localhost/18/insert_form.html a, 


Text to Add 


Insert Record 


图 18-1 用 来 添加 一 条 记录 的 HITML 表 单 


在 Text to Add 字 上 段 输 入 一 个 字符 串 ， 如 图 18-2 所 示 。 


fa] Record Insertion Form 


€ > C |O http://localhost/18/insert_form.html A, 


Text to Add: 
Little Bobby Tables 


Insert Record 


图 18-2 ” 表 蛙 字段 中 输入 的 文本 





最 后 ， 点 击 Insert Record 按 钮 来 执行 insert.php 肢 本， 并 且 插 入 记 
录 。 如 朵 成功 ， 将 会 看 到 如 图 18-3 所 示 的 结 





[=] localhost/16/insert.php 


他 C © http://localhost/18/insert.php aS 


A record has been mserted. 


图 18-3 ”记录 已 经 成 功 添加 


为 了 验证 我 们 的 工作 ， 可 以 使 用 MySQL 命 令 行 界面 来 得 看 表 中 的 
记录 。 


SELECT * FROM testTable; 


输出 应 该 如 下 所 示 : 


ee A aN oe, Hees 2. Ste 十 
id | testField | 
Oe ee eben Se + 
| 1 | some value | 
| 2 | Little Bobby Tables | 
Dacia ers area eee Ge + 
2 rows in set (6.00 sec) 


接 下 来 ， 我 们 将 学 习 如 何 使 用 PHP 获 取 和 格式 化 结果 ， 而 不 只 是 通 


过 命令 行 。 


18.3.3 ”使 用 PHP 获 取 数 据 


由 于 已 经 在 testTable 表 中 有 了 一 些 数据 行 ， 我 们 可 以 编号 一 个 PHP 
脚本 来 获取 这 些 数据 。 从 SQL 基础 知识 开始 ， 束 介绍 了 编写 一 个 脚本 来 
执行 一 条 SELECT 俘 询 语 名 。 但 是 我 们 这 里 的 SELECT 语 句 个 要 产生 太 
多 的 结果 数据 ， 我 们 只 要 得 到 行 数 ， 要 做 到 这 一 点 ， 束 必须 使 用 
mysdli_ num_rows0O 函 数 《〈 参 见 程序 清单 18.7 中 的 第 12 行 ) 。 





程序 清单 18.7 获取 数据 的 脚本 


1: <?php 

2: mysqli = mysqli _connect({"localhost", "joeuser", Somepass ， "testDB"); 
3: 

4: if (mysqli connect _errno({)) { 

5: printf( "Connect failed: %s\n", mysqli connect _error()); 

6: exit(); 

7: } else { 

8: $sql = "SELECT * FROM testTable"; 

9 : $res = mysqli query($mysqli, $sql); 

10: 

11: if ($res) { 

is $number of rows = mysqli _num_rows($res) ; 

13: printf({"Result set has %d rows.\n", $number of rows); 
14: } else { 

bee printf{"Could not retrieve records: %s\n", mysqli error($mysqli)); 
16: } 

TEs 

18: mysqli free _result($res) ; 

19: mysqli _close({$mysqli) ; 

20: } 

21: g> 


FEI 7 RAS GRAF Acount.php, FFE PE Websks as HCH Ase Fk, 
I a eS Webik saa ville. KSA AWU RIEU CERE, Pas 
将 根据 你 插入 到 表 中 的 记录 的 数目 而 有 所 不 同 ) 。 


Result set has 4 rows, 


第 12 行 使 用 mysqli_num_rows0 函 数 来 获取 结果 集 ($res) 中 的 行 


数 ， 然 后 ， 将 这 个 值 放 入 到 一 个 名 为 $gnumber_ of rows 的 变量 中 。 第 13 
行 在 浏览 器 中 显示 这 个 数字 ， 这 个 数字 等 于 你 在 测试 的 时 候 所 插入 的 记 
RAYA A o 
提示 : 
这 上 段 程 序 中 有 一 个 前 面 程序 没有 用 到 的 新 函数 ， 即 第 18 行 使 用 的 mysgli_free_result() 也 | 


数 。 在 使 用 mysqli_close() 函 数 关 闭 连 接 之 前 使 用 mysgli_free_result() 函 数 ， 这 确保 了 释放 与 查 
询 和 结果 相关 的 所 有 内 存 以 供 其 他 脚本 使 用 。 


既然 知道 了 表 中 有 一 些 记 录 《根据 输出 来 看 ， 征 4 条 记录 ) ， 可 以 
想象 并 获取 这 些 记 录 的 实际 内 容 。 我 们 可 以 用 几 种 方法 来 做 到 这 一 点 ， 
但 是 ， 了 最 简单 的 方法 束 是 获取 每 一 行 构成 一 个 数组 。 

我 们 将 使 用 一 条 while 语 句 来 过 历 结 果 集 中 的 每 条 记录 ， 把 每 个 字 
段 的 值 放 入 到 一 个 特定 的 变量 中 ， 人 然后 在 屏 科 上 显示 结果 。 
mysqli_fetch_array() 的 语法 如 下 。 


$newArray = mysqli fetch array{$result set}; 


下 面 使 用 程序 清单 18.8 中 的 示例 脚本 。 


程序 清单 18.8 ”获取 数据 并 显示 结果 的 一 个 脚本 


ON Oar, Wh 一 


<?php 
$mysqli = mysqli _connect("localhost", "joeuser", "somepass", "testDB"); 


if (mysqli connect _errno(}) { 
printf("Connect failed: %s\n", mysqli_connect_error()); 


exit(); 

} else { 
$sql = "SELECT * FROM testTable'"; 
$res = mysqli_query($mysqli, $sql); 


if ($res) { 
while ($newArray = mysqli fetch array($res, MYSQLI ASSOC)) { 
$id = $newArray['id']; 
$testField = $newArray[ 'testField']; 
echo "The ID is ".$id." and the text is: ".$testField."<br/>"; 
} 
} else { 
printf("Could not retrieve records: %s\n", mysqli error({$mysqli)); 


f 


mysqli free result($res}; 
mysqli_close($mysqli}; 


把 这 个 脚本 保存 为 select.php， 并 帮 营 在 Web 服 务 占 文档 目录 下 ， 然 


后 通过 Web 浏 虹 融 来 访问 它 。 你 将 会 看 到 ， 对 于 输入 到 testTable 中 有 的 每 
一 条 记录 都 将 有 一 条 消息 ， 如 图 18-4 所 示 。 这 些 消息 在 第 12 行 到 第 15 行 
的 While 循环 中 创建 。 


[5] localhost/16/select.php 
a CG © http://localhost/18/select.ohp ` 
The ID is 1 and the text is: some value 
The ID ts 2 and the text is: Little Bobby Tables 
The ID is $ and the text is: Here's more text! 
The ID is 6 and the text is: I really like cheese. 


图 18-4 从 MySQL 选 取 记 录 


基本 上 上 ， 我 们 可 以 仅仅 使 用 4 个 或 5 个 MySQLi 函 数 来 创建 一 个 完整 
的 数据 库 驱 动 的 应 用 程序 。 本 和 章 对 于 使 用 PHP 和 MySQL 只 是 浅 答 辑 目 
PHP 中 还 有 更 多 的 MySQLIi 函 数 。 


18.3.4 ”PHP 中 其 他 的 MySQL 消 数 


Pili #1002 4 FY H AY FE A MySQL A 2 
图 数 中 的 大 多 数 只 是 改变 了 获取 数据 的 方法 ， 或 改变 了 用 来 搜集 所 
吉 构 的 信息 。 在 整 本 书 中 ， 特 别 是 在 稍 后 和 项 目 相 天 的 各 章 
中 ， 你 将 逐渐 认识 到 PHP 中 更 多 的 MySQL 专 用 函数 。 然 而 ， 要 获取 函数 
的 完整 列表 以 及 实际 的 例子 ， 请 访问 位 于 http://www.php.net/mysqli 的 
PHP 手 册 的 MySQLi 部 分 。 


18.4 小结 


使 用 PHP 和 和 MySQL 来 创建 动态 的 、 数 据 库 驱 动 的 Web 站 点 只 是 小 采 
— he. FAS 了 ，PHP 函 数 基 本 上 古 通 往 数 据 库 服务 右 的 一 个 关口 ， 用 
MySQL 命 令 行 界面 输入 的 任何 内 容 ， 都 可 以 使 用 mysgli_gquery0 孙 数 输 
入 。 我 们 还 学 习 了 从 一 个 表单 接 党 用 户 和 输入 的 时 候 如 何 避 免 $SQL 注 入 。 


要 使 用 PHP 连 接 到 MySQL， 你 需要 知道 MySQL 用 户 名 、 密 但 和 数 
据 库 名 。 一 旦 连接 上 ， 可 以 使 用 mysqli_gquery0 函 数 执 行 标准 的 SQL 命 
令 。 如 果 已 经 执行 了 一 条 SELECT 命 令 ， 可 以 使 用 mysqli_num_rows() 米 
计算 结果 集中 返回 的 记录 的 条 数 。 如 果 想 要 显示 得 到 的 数据 ， 可 以 在 一 
个 循环 中 使 用 mysqli_fetch_array() 来 获取 所 有 记录 并 将 它们 显示 到 屏 和 项 
上 。 


18.5 Q&A 


Q: 能 够 在 一 个 应 用 程序 中 同时 使 用 mysqlL_* 和 mysqli_* 函 数 
IE ? 


A: 如 条 PHP 安 闭 后 对 两 个 库 都 文 持 ， 我 们 可 以 使 用 任何 一 组 函数 
来 和 MySQL 通 信 。 然 而 ， 注 意 ， 如 果 对 MySQL 4.1.3 以 后 的 版 本 使 用 
mysql * 函 数组 ， 可 能 不 能 够 访问 某 些 新 的 功能 。 此 外 ， 如 果 在 整个 应 
用 程序 中 采用 不 一 致 的 用 法 ， 维 持 和 维护 应 用 程序 将 会 耗费 更 多 时 间 并 
且 效 果 也 不 理想 。 


18.6 ”实践 练习 


实践 练习 十 设 计 用 来 帮助 你 预料 可 能 的 问题 、 复 习 已 经 学 过 的 知 
识 ， 并 且 开 始 把 知识 用 于 实践 。 


练习 题 


1. 用 来 进行 PHP 和 MySQL 之 间 的 连接 的 主要 函数 是 什么 ”需要 哪 
些 信 息 ? 


.哪个 PHP 函 数 获取 一 个 MySQL 错 误 消息 的 文本 ? 


.哪个 PHP 函 数 用 来 计算 一 个 结 来 集中 的 记录 的 条 数 ? 


Fa 
oy 


1. mysqli_connect() KAE] 4 —S SIMySQLAI#, mE 
FAP ZAMR, 


2. mysqli_error() K ZUR E — Kh MySQL fa RÄ o 


3. mysqli_num_rows() 函 数 计算 结果 集中 的 记录 数目 。 


EA wel 


1. 使 用 一 个 HTML 表单 和 PHP 脚 本 ， 创 建 一 个 表 ， 其 字段 包含 了 一 
个 人 的 姓氏 和 名 字 。 创 建 男 外 一 段 脚 本 ， 来 同 表 中 添加 记录 。 


2. 一 旦 表 中 有 了 记录 ， 创 建 一 段 PHP 脚 本 获取 记录 ， 并 且 按 照 姓 
氏 的 字母 顺序 显示 这 些 记 录 。 


eS 5isa = Z£ ATA A 


党 理 一 个 简单 的 邮件 列表 
创建 一 个 在 线 地 址 竹 

创建 一 个 徐 单 的 讨论 论坛 
创建 一 个 在 线 商店 
创建 一 个 购物 车 机 制 
创建 一 个 简单 的 日 历 
限制 对 应 用 程序 的 访问 
记录 并 监视 Web 服 务 器 活动 
应 用 程序 本 地 化 


使 用 XML 


FAME HES fe] FLA MB AE Be 


在 本 革 中 ， 你 将 学 习 到 : 


。 如 何 创 建 一 个 订阅 / 退 订 表 里 和 脚本 。 
。 MAA) A AIA IE EEE — A H it o 
。 WE EREK US A AK 


本 章 提 供 了 一 个 自己 动手 的 小 项 目 ， 后 续 章 节 还 将 有 类 似 的 项 目 ， 
这 些 项 目 设计 用 来 把 我 们 APSE 到 一 起 。 本 草 ， 我 们 
将 学 习 创 建 一 个 可 管 人 这 个 列表 可 以 用 来 发 送 新 闻 
组 邮件 或 者 其 他 任何 东西 ， 这 些 东 西 是 要 发 送 给 数据 库 中 的 一 个 邮件 地 
址 列表 的 。 


和 本 书 中 有 所 有 小 的 示例 项 目 一 样 ， 这 些 项 目 可 能 和 你 想 要 用 目 己 的 
A 构建 的 项 目 不 完 全 相同 。 然 而 ， 我 再 怎么 强调 一 点 也 
不 过 分 ， 这 就 是 这 个 项 目 以 及 其 他 项 目 中 展示 的 概念 和 示例 ， 与 你 开发 
任何 使 用 CRUD 功 能 (创建 、 读 取 、 更 新 和 删除 〉 的 应 用 程序 时 所 过 到 
概念 和 示例 是 类 似 的 。 


我 们 在 本 重 中 用 到 的 邮件 机 制 并 不 是 用 来 将 代 邮 件 列 表 软 件 的 ， 后 
者 是 专门 设计 来 群 肥 邮件 的 。 我 们 在 本 章 中 构建 的 这 种 系统 只 能 用 于 小 
型 列表 ， 即 小 于 数 白 个 邮件 地 址 的 列表 。 


19.1 开发 订阅 机 制 


我 们 在 前 面 的 各 章 中 所 学 习 的 ， 都 是 创建 任何 产品 所 需 的 最 重要 的 
方面 。 在 这 个 例子 中 ， 攻 谨 我 们 的 订阅 机 制 所 需要 的 如 下 元 系 。 


e 一 个 保存 Email 地 址 的 表 。 
o 用 户 添 加 或 删除 Email 地 址 的 方法 。 
。 及 达 消 县 的 表单 和 脚本 。 


下 面 各 节 分 别 介 绍 每 个 项 目 元 素 。 
19.1.1 创建 subscribers 表 


在 subscribers 表 中 ， 我 们 真 的 只 需要 一 个 字段 来 存储 用 户 的 Email 地 
址 。 然 而 ， 我 们 应 该 有 一 个 ID 字段 ， 目 的 只 是 为 了 维护 表 之 间 的 一 臻 
性 ， 并 且 也 因为 在 where 子 句 中 引用 一 个 ID 比 引 用 一 个 长 长 的 Email 地 址 
要 简单 得 多 。 因 此 ， 在 这 个 例子 中 ， 创 建 表 的 SQL 语句 将 会 如 下 所 示 。 


CREATE TABLE subscribers { 

id INT NOT NULL PRIMARY KEY AUTO INCREMENT, 

email VARCHAR (15@) UNIQUE NOT NULL 
); 

注意 在 Email 的 字段 定义 中 UNIQUE 的 使 用 。 这 意味 着 ， 尽 管 id 是 主 
键 ， 但 是 也 不 允许 在 Email 字段 出 现 重复 。Email 字 段 也 是 唯一 键 ， 但 是 
id 是 主键 。 


通过 命令 行 登 录 到 MySQL 并 执行 这 个 查询 。 在 创建 了 表 之 后 ， 执 
行 一 个 DESC 或 DESCRIBE 查 询 来 验证 表 已 经 按照 你 的 指定 创建 了 ， 示 


例如 下 。 


mysql> as subscribers; 


和 Passas s Paaa Pearse mian 下 十 
| Field Type | Null | Key | Default | Extra | 
easier esr en oe Tepe a es yn eae E + 
| id i int{11) | NO | PRI | NULL | auto increment | 
| email | varchar(15®) | NO | UNI | NULL | | 
ee ne mata, SRN Se is sursa Forani anet ee ee + 


2 rows in set (0.00 sec} 


DEIR TE BU EAA STK, RITI NEEE IBA H EY ee FT A 
ASS 


19.1.2 ”为 共同 函数 创建 一 个 包含 文件 


尺 官 这 个 过 程 中 只 有 两 个 脚本 ， 它 们 之 中 还 是 有 一 些 共同 函数 ， 即 
数据 库 连 接 信息 。 要 让 脚本 在 这 种 情况 下 更 为 精简 ， 把 共同 函数 或 代码 
段 取出 来 ， 并 放 到 一 个 文件 中 ， 然 后 通过 我 们 在 第 13 章 学 习 的 include 语 
句 将 文件 包 合 到 其 他 脚本 中 。 程 序 清单 19.1 包 含 了 本 章 的 脚本 所 共 盏 的 
代码 。 


程序 清单 19.1 包含 文件 中 的 共同 函数 


1: <?php 

2: // function to connect to database 

3: function doDB() 1 

4: global $mysqli; 

aa 

6: /{/connect to server and select database 

了 $mysqli = mysqli connect({"localhost", "joeuser", 
B: “Somepass ， “testDB"}); 

9: 

10: [{if connection fails, stop script execution 
Tis if (mysqli_connect_errno()) { 

q2: printf("Connect failed: %s\n", mysqli_connect_error()); 
(3; exit(); 

14: } 

15: } 


16: // function to check email address 
17: function emailChecker($email) { 


16: global $mysqli, $safe_email, $check res; 

19: 

20: / {check that email is not already in list 

21% $safe email = mysqli_real_escape string($mysqli, $email); 
22: $check sql = "SELECT id FROM SUBSCRIBERS 

20% WHERE email = '".$safe_email."'"; 

24: $check res = mysqli query($mysqli, $check sql) 

2 or die(mysqli error{$mysqli)})); 

26: 4 

ot Sn 


第 3 到 15 行 建立 了 一 个 函数 doDBO， 这 是 一 个 简单 的 数据 库 连 接 函 
数 。 如 果 没 有 进行 连接 ， 在 调用 这 个 函数 的 时 候 ， 脚 本 将 会 退出 ; F 
则 ， 它 将 让 $mysqli 的 值 可 供 脚 本 其 他 部 分 使 用 。 


第 17 行 到 第 26 行 定义 了 一 个 名 为 emailChecker0 的 函数 ， 它 接受 一 
个 输入 并 返回 一 个 输出， 就 像 大 多 数 图 数 所 做 的 那样 。 当 我 们 看 到 程序 
清单 19.2 的 时 候 ， 将 在 脚本 中 见 到 这 个 函数 。 


把 这 个 文件 保存 为 ch19_include.php 并 将 其 放置 到 Web 服 务 器 上 。 在 
程序 清单 19.2 中 ， 我 们 将 会 看 到 ， 在 需要 的 时 候 如 何 把 这 个 文件 包含 到 
脚本 中 。 


19.1.3 ”创建 订阅 表单 


订阅 表单 实际 上 是 一 个 名 为 manage.php 的 一 体 化 表单 和 脚本 ， 它 会 
处 理 订 阅 和 退 订 请 求 。 程 序 清单 19.2 给 出 了 manage.php 的 代码 ， 它 使 用 
几 个 用 户 定 义 的 函数 来 避免 重复 的 代码 ， 并 且 由 此 开始 考虑 创建 目 己 的 
函数 。 代 人 码 看 上 去 很 长 ， 但 后 面 将 逐 行 说 明 ， 并 且 这 上 段 代 人 码 的 主要 部 分 
ÆHTMLR E, TALC RIFE! 


程序 清单 19.2 ”使 用 manage.php 订 疯 和 退 订 


Ona apoan — 


ce) 


<?php 
include ‘ch19 include.php' ; 
//determine if they need to see the form or not 
if (I$ POSTY 4 
//they need to see the form, so create form block 
$display_block = <<<END OF BLOCK 
<form method="POST" action="$ SERVER[PHP SELF]"> 


<p><label for="email">Your E-Mail Address:</label><br/> 
<input type="email" id="email" name="email" 
size="40" maxlength="150" /></p> 


<fieldset> 

<legend>Action:</legend><br/> 

<input type="radio’ id="action sub" name="action" 
value="Sub" checked /> 

<label for="action_sub">subscribe</label><br/> 

<input type="radio" id="action unsub" name="action" 
Value="unsub" /> 

<label for="action unsub">unsubscribe</label> 

</fieldset> 


<button type="submit" name="submit" value="submit">Submit</button> 
</form> 
END OF _ BLOCK; 
} else if ({$ POST} && ($ POST['action'] == "sub"}) { 
//trying to subscribe; validate email address 
if ($ POST[ email ] == "") { 
header("Location: manage.php"); 
exit; 
} else { 
//connect to database 
doDB{); 


į {check that email is in list 
emailChecker({($ POST['email']); 


//get number of results and do action 
if {mysqli num rows{$check res) < 1) { 
{//free result 
mysqli free result($check res}; 


//add record 
$add sql = "INSERT INTO subscribers (email) 
VALUES(‘".$safe email."')"; 


100: 
101: 
102: 
103: 


$add res = mysqli query($mysqli, $add sql) 
or die(mysqli_error($mysqli) ) ; 
$display block = "<p>Thanks for signing up!</p>"; 


//close connection to MySQL 
mysqli_close($mysqli); 
} else { 
//print failure message 
$display block = "<p>You're already subscribed!</p>"; 
} 
} 
} else if ({$_POST) && ($_POST[‘'action'] == “unsub")) { 
//trying to unsubscribe; validate email address 
if ($ POST['email'] == "") { 
header("Location: manage.php") ; 
exit; 
} else { 
//connect to database 
doDB( ) ; 


//check that email is in list 
emailChecker($ _POST['email']); 


//get number of results and do action 
if {mysqli num rows{$check res) < 1) { 
//free result 
mysqli free _result($check_res); 


//print failure message 
$display_block = "<p>Couldn't find your address!</p> 
<p>No action was taken.</p>"; 
} else { 
//get value of ID from result 
while ($row = mysqli _fetch_array($check_res)) { 
$id = $row['id']; 
} 


//unsubscribe the address 
$del_sql = "DELETE FROM subscribers 
WHERE id = ".$id; 
$del res = mysqli query({$mysgqli, $del_sql) 
or die(mysqli error($mysqli) ); 
$display_block = "<p>You're unsubscribed!</p>"; 
} 
mysqli _close({$mysql1li) ; 


} 

?> 

<!DOCTYPE html> 

<html> 

<head> 

<title>Subscribe/Unsubscribe to a Mailing List</title> 
</head> 

<body> 

<h1>Subscribe/Unsubscribe to a Mailing List</h1> 
<?php echo "$display block"; ?> 

</body> 

</html> 


程序 清单 19.2 可 能 有 点 长 ， 但 是 它 并 不 复杂 。 实 际 上 ， 如 果 不 是 把 
用 户 定义 的 函数 放 在 了 ch19_include.php 中 并 在 这 段 脚 本 的 第 2 行 包含 它 
的 话 ， 程 序 清 单 19.2 可 能 会 更 长 。 


此 4 行 开 始 了 脚本 的 主要 逻辑 。 由 于 这 个 脚本 执行 了 几 个 操作 ， 我 
们 高 要 自 和 完 确定 它 当 前 答 试 的 是 哪 一 个 操作 。 如 末 $_POST 的 结果 为 
false, FAH PBA pete, A, FTAA Pika 
单 。 


第 6 行 到 第 25 行 ， 使 用 heredoc 格 式 将 一 个 字符 串 存 储 到 
$display_block 变 量 中 ， 创 建 了 订阅 / 退 订 表单 。 在 heredoc 格 式 中 ， 字 符 
串 分 隔 符 可 能 是 <<< 后 面 的 任何 字符 串 标 识 待 ， 只 要 了 最 终 的 结束 标识 符 
丫 独 一 行驶 行 了 了， 正如 你 在 这 个 示例 的 第 25 行 中 所 见 到 的 那样 。 


提示 : 





可 以 通过 位 于 http:// www.php.net/ manual/en/ language. types. string. php 的 PHP 手 册 ， 了 解 
关于 heredoc 和 其 他 字符 串 格 式 的 更 多 信息 。 





在 这 个 表单 中 ， 我 们 使 用 $_SERVER[PHP_SELF] 作 为 动作 (第 7 
行 ) ， 然 后 ， 创 建 了 一 个 名 为 email 的 用 于 用 户 的 邮件 地 址 的 文本 字段 
(第 9 行 到 第 11 行 ) ， 并 且 设 置 了 一 组 单 选 按钮 〈 第 13 行 到 第 21 行 ) 来 
表示 想 要 执行 的 任务 。 在 字符 串 创建 的 最 后 ， 脚 本 跳出 了 if...else 结 构 ， 
跳 到 了 第 101 行 ， 并 且 显 示 变 量 $display_block 中 存储 的 HTML。 这 个 表 
单 显示 如 图 19-1 所 示 。 





© Subscribe/Unsubscribe to a 


€ > C O hito://localhost/19/manage.php Ki A *® 


Subscribe/Unsubscribe to a Mailing List 


Your Email Address: 


— Action ——— SnD Se NE ih nh Dane a ad a ined base Pas aint eS tt ia rn ie sn oe 


© subscribe 
© wosubscribe 


图 19-1 订阅 / 退 订 表单 


回 到 if...else 结 构 ， 如 果 $_POST 的 结果 为 true， 我 们 需要 做 一 些 事 
情 。 这 有 两 种 可 能 : 对 于 表单 中 提供 的 Email 地 址 的 订阅 或 退 订 动作 。 
我 们 需要 看 单 选 按钮 组 中 的 $ POST['action] 的 值 来 确定 采取 哪个 操作 。 


在 第 26 行 ， 如 末 $_POST 的 结束 为 tue 并 且 $_POST[action] 的 伸 
为 “sub”， 我 们 知道 用 户 要 订 疝 。 为 了 订阅 ， 用 户 需要 一 个 Email 地 址 ， 
因此 ， 在 第 28 行 到 第 30 行 检查 这 个 地 址 。 如 果 没 有 地 址 ， 用 户 将 会 重 定 
回回 到 该 表单 。 


然而 ， 如 果 有 地 址 ， 在 第 33 行 调用 doDBO 函 数 〈 存 储 在 
ch19_include.php 中 ) 来 连接 数据 库 ， 以 便 可 以 及 送 租 询 。 在 第 36 行 ， 我 
们 调用 第 二 个 用 户 定 义 函 数 emailChecker()。 这 个 函数 接受 一 个 输入 
(在 本 例 中 是 $_ POST [email]) 并 处 理 它 。 如 果 回 头 看 看 程序 清单 19.1 
的 第 21 行 到 第 25 行 ， 会 看 到 emailChecker() 函 数 中 的 代码 执行 了 一 个 查 


询 ， 试 图 针对 包含 传递 给 函数 的 Email 地 址 的 记录 ， 在 subscribers 表 中 找 
到 一 个 id 值 。 这 个 函数 随后 返回 名 为 $check_res 的 结果 集 ， 以 供 在 调用 
的 脚本 中 使 用 。 


提示 : 


注意 ， 程 序 清单 19.1 中 的 两 个 用 户 定 义 函 数 的 开始 ， 都 有 全 局 变量 的 定义 。 这 些 变 量 由 整 
个 脚本 共享 ， 并 且 为 此 声明 为 全 局 的 。 





跳 到 下 面 的 第 39 行 ， 看 看 $check_res 变 量 是 如 何 使 用 的 : 把 
$check_res 变 量 所 引用 的 记录 数目 计算 出 来 ， 以 确定 Email 地 址 是 售 已 经 
在 表 中 存在 。 如 果 行 数 小 于 1， 表 明 这 个 地 址 没有 在 列表 中 ， 并 且 它 将 
会 航 添 加 到 列表 中 。 记 录 添 加 之 后 ， 啊 应 在 第 44 行 到 第 48 行 存储 。 而 失 
败 消息 《如 条 地 址 已 经 在 表 中 存在 ) 在 第 54 行 存储 。 此 时 ， 脚 本 跳出 
if...else t, [a] POLI EIER101{7, MZANHTML. Fe ee MK 
功能 。 


如 果 $_POST 的 结果 为 true 并 且 $_POST['action] 变 量 的 值 是 “<unsub”， 
就 会 出 现 输入 的 最 后 一 种 组 合 。 在 这 种 情况 下 ， 用 户 试 图 退 订 。 为 了 退 
订 ， 需 要 一 个 已 有 的 Email 地 址 ， 因 此 ， 我 们 在 第 59 行 到 第 61 行 检查 
它 。 如 果 没 有 给 出 地 址 ， 表 单 会 回 传 给 用 户 。 


如 果 有 地 址 ， 我 们 在 第 64 行 调用 doDB() 孙 数 连 接 到 数据 库 。 然 后 ， 
在 第 67 行 调用 emailChecker() 疯 数 ， 它 将 再 次 返回 结果 集 $check_res。 在 
第 70 行 ， 计 算 结 果 和 集中 的 记录 数 ， 从 而 来 确定 Email 地 址 是 耕 已 经 存在 
于 表 中 。 如 果 行 数值 小 于 1， 地 址 没有 在 列表 中 ， 这 样 将 无 法 退 订 。 


在 这 个 例子 中 ， 啊 应 消 恩 在 第 75 行 到 第 76 行 存储 。 然 而 ， 如 来 行 数 
不 小 于 1， 用 户 将 农 退 订 《 记 录 删 除 ) 并 且 在 第 84 行 到 第 88 行 存储 咖 
图 19-2 到 图 19-5 给 出 了 这 个 脚本 根据 选择 的 操作 和 数据 库 中 的 Email 


HODES TRS PI AE 
El 


a) Subscribe/Unsubscribe to = 
5 C | © http://localhost/19/manage.php KH KE 


Subscribe/Unsubscribe to a Mailing List 


Thanks for signing up! 


图 19-2 成功 订阅 
| |i 


© Subscribe/Unsubscribe to a 
& @ | © http://localhost/19/manage.php Ki KE 


Subscribe/Unsubscribe to a Mailing List 


You're already subscribed! 


图 19-3 ”订阅 失败 


-| i 
a) Subscribe/Unsubscribe to a 
€ C | © http://localhost/19/manage.php KM mA ®y 


Subscribe/Unsubscribe to a Mailing List 


You're unsubscnbed! 


图 19-4” 成功 退 订 


() Subscribe/Unsubscribe to a 


& @ | © http://localhost/19/manage.php Ki A ww 


Subscribe/Unsubscribe to a Mailing List 


Couldn't find your address! 


No action was taken. 


图 19-5” 退 订 不 成 功 


接 下 来 ， 我 们 将 创建 给 每 个 订阅 者 发 送 Email 的 表单 和 脚本 。 


19.2 开发 邮件 发 送 机 制 


准备 好 了 订阅 机 制 ， 我 们 可 以 在 一 个 脚本 创建 一 个 基本 的 表单 ， 这 
个 脚本 从 表单 提取 内 容 ， 并 且 将 其 发 送 给 subscribers 表 中 的 每 个 地 址 。 
下 面 是 男 一 个 完整 的 脚本 ， 名 为 sendmymail.php， 程 序 清单 19.3 给 出 其 
内 容 。 


提示 : 











在 壬 试 使 用 本 市 中 的 脚本 之 前 ， 确 保 已 经 阅读 了 第 11 革 中 天 于 在 php.ini 文 件 中 进行 配置 的 
部 分 ， 这 是 发 送 邮 件 所 必需 的 。 


程序 清单 19.3 ” 回 订 阅 者 列表 发 送 邮件 


1: <?php 

2: include 'chi9 include.php' ; 

3: if (I$ POST) { 

4: {/fhaven't seen the form, so display it 

5: $display block = <<<END OF BLOCK 

6: <form method="POST" action="$ SERVER[PHP SELF] "> 

Fi 

8: <p><label for="subject">Subject:</label><br/> 

9 : <input type="text" id="subject" name="subject" size="4@" /></p> 

1@: 

11% <p><label for="message">Mail Body:</label><br/> 

12: <textarea id="message" name="message" cols="50" 
rows='1@0'"></textarea></p> 

Ta: <button type="submit" name="submit" value="submit">Submit</button> 

14: </form> 


15: END_OF_BLOCK; 
16: } else if ($ POST) { 


Fae //want to send form, so check for required fields 


18: if (($ POST['subject'] == "") || ($ POST['message'] == "")) { 
19: header("Location: sendmymail.php"); 

20: exit; 

d i } 

at AR 

eas //connect to database 

24: doDB(); 

on 

26: if (mysqli _connect_errno{)) { 

2 //if connection fails, stop script execution 

28: printf ("Connect failed: %s\n", mysqli _connect_error()); 
29: exit(); 

30: } else { 

OT: //otherwise, get emails from subscribers list 

gos $sql = "SELECT email FROM subscribers"; 

33: $result = mysqli query($mysqli, $sql) 

34: or die(mysqli error($mysqli) ); 

oo 

36: //create a From: mailheader 

Si $mailheaders = "From: Your Mailing List 

38: <you@yourdomain.com>"; 

39; //loop through results and send mail 

40: while ($row = mysqli fetch array($result)) { 

41: set_time_limit(@); 

42: $email = $row[ 'email']; 

43: mail("$email", stripslashes($ POST['subject']}, 

44: Stripslashes($ POST[ 'message']), $mailheaders) ; 
45: $display block .= "newsletter sent to: ".$email. "<br/>"; 
46: } 

47: mysqli free result ($result); 

48: mysqli _close($mysql1li) ; 

49: } 

50: } 

51: ?> 

52: <!DOCTYPE html> 

53: <html> 

54: <head> 

55: <title>Sending a Newsletter</title> 

56: </head> 

57: <body> 


58: <hi>Send a Newsletter</h1> 
59: <?php echo $display block; ?> 
60: </body> 

61: </html> 


在 程序 清单 19.3 中 ， 用 户 定 义 的 函数 的 文件 在 第 2 行 航 包含 进来 。 
尽 害 在 这 个 文件 中 只 有 数据 库 连 接 函 数 会 用 到 ， 但 把 其 他 的 函数 包 合 在 


这 个 文件 中 也 无 伤 大 鸦 。 

脚本 的 主要 志和 辑 从 第 3 行 开 始 ， 在 那里 ， 我 们 确定 用 户 是 售 已 经 看 
到 表单 。 如 果 $_POST 变 量 为 false， 我 们 知道 用 户 还 没有 提交 表单 ， 
此 ， 我 们 必须 显示 页 面 。 

第 5 行 到 第 15 行 创建 了 辐 订 阅 者 列表 及 送 新 闻 组 邮件 的 表单 ， 它 使 
用 $_SERVER [PHP_SELF] 作 为 动作 《第 6 行 ) ， 创 建 一 个 叫做 subject 的 
文本 字段 来 表示 邮件 的 主题 ， 并 且 创 建 了 一 个 名 为 message 的 文本 域 来 
表示 要 发 这 的 信件 的 正文 。 

在 这 里 ， 脚 本 跳出 了 if...else 构 造 ， 并 且 HTML 显 示 出 来 。 所 显示 的 
表单 如 图 19-6 所 示 。 





Q Sending a Newsletter 
€ > C |O localhost/19/sendmymail.php 


Send a Newsletter 


Subject 


Mail Body- 


图 19-6 ” 友 送 大 批 邮 件 的 表单 


如 果 $_POST 的 结果 不 是 false， 脚 本 将 会 把 表单 友 运 到 subscribers 表 
中 的 邮件 地 址 。 和 在 我 们 发 送 消息 前 ， 我 们 必须 在 第 18 行 到 第 20 行 检查 来 
自 表 单 的 两 个 必需 项 目 : $_ POST['subject'] 和 $_POST[message]。 如 果 
这 两 个 项 目 中 有 一 个 不 存在 ， 用 户 将 会 再 次 重 定 加 表单 。 


如 采 所 需 项 目 具 备 了 ， 脚 本 移动 到 第 24 行 调用 数据 库 连 接 函 数 。 第 
33 行 执行 一 个 租 询 ， 从 subscribers 表 抓 取 所 有 的 Email 地 址 。 尽 管 这 些 纬 
果 没 有 顺序 ， 但 是 如 果 由 于 某 种 原因 我 们 想 要 按照 字母 顺序 送出 邮件 ， 
可 以 在 这 里 使 用 一 条 order by 子 句 。 


第 37 行 到 第 38 行 创建 了 一 个 From: 邮 件 标 头 ， 当 邮件 发 送 的 时 候 ， 
这 个 标 头 将 用 在 未 来 的 while 循 环 中 。 这 个 标 头 确保 了 邮件 看 上 去 像 是 
来 日 菏 一 个 人 而 不 是 一 台 机 人 右 ， 因 为 我 们 已 经 明确 地 在 这 个 字符 串 中 所 
供 了 一 个 值 。 开 始 于 第 40 行 的 while 循 环 ， 每 次 从 结果 集中 提取 一 个 邮 
件 地 址 。 在 第 41 行 ， 我 们 使 用 set_time_limitO) 函 数 把 时 间 限 制 设置 为 0 或 
者 “no limit”。 这 么 做 允许 脚本 运行 所 需要 的 那么 长 时 间 。 


提示 : 





由 于 程序 清单 19.3 中 的 所 有 脚本 多 次 执行 了 mail0 函 数 ， 它 确实 没有 考虑 到 实际 邮件 列表 
软件 中 的 排队 因素 ， 这 是 为 减轻 友 送 邮件 服务 器 负担 而 设计 的 。 使 用 set_time_limitO 并 不 会 绥 
解 这 一 负担 ， 它 只 是 在 时 间 可 能 超时 之 前 允许 脚本 继续 运行 。 








在 第 43 行 到 第 44 行 ， 使 用 mail0 函 数 发 送 邮 件 ， 插 入 了 来 自 相 应 的 
表单 的 值 。 第 45 行 癌 屏 幕 显 示 了 一 条 消息 ， 以 便 香 知 谁 应 该 接受 邮件 。 
在 图 19-7 和 图 19-8 中 ， 我 们 看 到 了 脚本 的 输出 。 


| Sending a Newsletter 
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Send a Newsletter | 





newsletter sent to: jane(@doe.com 
| newsletter sent to: jemeloni@ gmail.com 
newsletter sent to: john/@smuith.com 


















































图 19-7 ”邮件 已 经 发 送 


| @ This is a Test Mail - jcmeloni@gmail.com - Gmail - Google Chrome 


https://mail.google.com/mail/?ui=28&view=btop&tver=Lézqbeznst35&search=inboxBth=13563/010289b153"¢ 


This is a Test Mail ™ inbox x | 
m= Print all 


Your Mailing List youq@@yourdomain.com 5:27 PM (2 minutes ago) ~ |7 
to me [=] 


| like cheese. Do you? 


©. Click here to Reply or Forward 





图 19-8 ”邮件 安全 到 达 


19.3 ”小结 


在 这 个 动手 实验 的 一 章 中 ， 我 们 应 用 了 基本 的 PHP 和 MySQL 知 识 来 
创建 一 个 个 人 邮件 列表 。 包 括 数据 库 表 的 创建 、 订 阅 / 退 订 机 制 ， 以 及 
发 送 邮 件 的 表单 和 脚本 。 


19.4 Q&A 


Q: BAA] EAE HI 14 AR AS A HY Gi TE? 


A: 除了 深入 理解 打包 了 的 邮件 列表 软件 ， 我 们 可 以 通过 一 个 
socket 连 接 统 过 mail() 函 数 并 和 直接 和 SMTP 服 务 右 对 十。 这 样 的 例子 在 
PHP Ht Hy fsockopen() AZt BJ LIRR 2 (http://www.php. net/fsockopen ), 
FEIT R YR Ete ay DFR FI. 


Q: Me VWF AA xe BUMS E ? 
A: AlfE4A]Email PE OANA xe Fa HG AS Be PIR IA DUIS HY BB 


些 ) ， 啊 应 消息 去 往 我 们 在 From: 或 Reply-to 邮 件 标 头 中 指定 的 任何 地 
址 。 


19.5 ”实践 练习 


这 个 实践 练习 设计 用 来 帮助 你 预测 可 能 的 问题 、 复 习 已 经 学 过 的 知 
识 ， 并 且 开 始 把 知识 用 于 实践 。 


问答 题 
1. 什么 函数 发 送 邮 件 ? 
2. 为 什么 在 程序 清单 19.1 中 $mysqli 指 定 为 一 个 全 局 变量 。 


3. 什么 函数 调用 导致 脚本 执行 它 所 需 的 那么 长 时 间 ? 


Fa 
oy 


1. 这 不 是 一 个 很 难 的 问题 ，mail(0) 函 数 可 实现 。 


2. 因为 变量 $mysqli 在 一 个 函数 中 创建 并 赋值 ， 而 这 个 函数 被 包 侣 
在 供 其 他 函数 使 用 的 一 段 脚 本 中 ， 因 此 ， 访 变量 必须 声明 为 全 局 的 ， 以 
确 你 在 创建 它 的 函数 之 外 也 能 使 用 它 。 


3. set time limit(0)。 


BA Wel 


1， 修 改 manage.php 脚 本 ， 将 用 户 的 Email 作为 任何 操作 的 啊 应 消 县 
HI AB OP TI SAB o 


2. 修改 sendmymail.php 脚 本 ， 以 添加 额外 的 表单 字段 ， am 
IA BP) 4a ep eH ie. WE, AREER, 
字符 串 必 须 串 成 一 个 消息 字符 串 ， 并 访 字 人 符 串 发 送 给 mailO0 函 数 。 


第 20 革 ”创建 一 个 在 线 地 址 竹 


ERER, MÄ FI: 


。 TA AS FE DL Te BW EEK BRE 
。 如 何 创建 表单 和 脚本 来 添加 和 删除 地 址 每 中 的 记录 。 
。 如 何 创建 表单 和 脚本 来 浏览 记录 。 


在 这 个 动手 实验 的 章节 里 ， 项 目 将 创建 一 个 可 管理 的 和 在线 地 址 鲜 。 
我 们 将 学 习 创 建 相关 的 数据 库 表 的 方法 ， 以 及 创建 表 蛙 和 脚本 来 添加 、 
删除 和 浏览 数据 库 记 录 的 方法 。 


这 些 基 本 概念 ， 征 开 友 任何 使 用 CRUD 功 能 《创建 、 谈 取 、 更 新 和 
删除 ) 的 应 用 程序 的 基础 ， 这 一 点 和 很 多 Web 应 用 程序 都 是 一 样 的 。 


20.1 规划 和 创建 数据 库 表 


当 我 们 考虑 一 个 地 址 筹 的 时 候 ， 有 些 明 显 的 字段 会 涌 入 脑海 ， 名 
字 、 地 址 、 电 话 号 码 、 邮 件 地 址 等 等 。 然 而 ， 如 果 我 们 查看 纸 质 的 地 址 
短 ， 可 能 会 注意 到 一 个 人 有 多 个 条 目 。 可 能 这 个 人 有 3 个 电话 号 码 或 者 
两 个 Email 地 址 等 等 ， 或 者 任何 与 原始 模板 不 相符 的 地 方 。 在 我 们 的 在 
线 地 址 短 中 ， 一 组 相关 的 表 将 协助 缓解 信息 的 元 余 和 重复 ， 并 且 人 允许 我 
们 以 一 致 的 视图 来 显示 所 有 信息 。 


表 20-1 给 出 了 用 于 在 线 地 址 短 的 示例 表 及 其 字段 名 ， 我 们 将 使 用 真 
下 的 SQL 语 句 来 创建 这 些 表 ， 但 自 和 完 应 访 看 看 这 些 信 息 并 答 试 看 看 现 有 
的 天 系 。 目 己 思 考 一 下 哪个 字段 应 该 是 主键 或 唯一 键 。 


表 20-1 地 址 注 表 和 字段 名 


id, date_added, date_modified, {f name, | name 


id, master_id, date_added, date_modified, address, city, state, zipcode, 


type 


id, master_id, date_added, date_modified, tel_number, type 


id, master_id, date_added, date_modified, fax_number, type 





email id, master_id, date_added, date_modified, email, type 


id, master_id, date_added, date_modified, note 





注意 和 日 期 相关 的 字段 的 用 法 ， 每 个 表 中 都 有 一 个 date_added 和 
date_Modified 字 段 。 这 个 字段 将 帮助 你 维护 数据 ， 我 们 可 能 在 某 个 时 刻 
要 执行 一 个 查询 ， 把 多 少 个 月 之 前 或 多 少年 之 前 的 所 有 记录 删除 挥 ， 或 
者 把 在 菏 段 时 间 内 没有 更 新 过 的 所 有 记录 删除 挥 。 


正如 将 在 如 下 的 SQL 语句 中 见 到 的 ， 除 了 ID 以 及 和 日 期 相关 的 字 
Ez, master namek MÚS ATZ, Af name 和 1] name, IRZ 
名 和 姓 ，id 字 段 是 主键 。 不 需要 其 他 的 字段 作为 主键 或 唯一 键 ， 除 非 我 
们 真 的 需要 限制 地 址 短 只 能 有 一 个 John Smith、 一 个 Mary Jones 等 。 


提示 : 











下 面 的 语句 中 的 文本 字段 的 字段 长 度 是 任意 的 ， 你 可 以 让 它们 尽 可 能 地 短 ， 只 要 在 字段 
类 型 允许 的 定义 内 。 


下 面 的 SQL 语句 创建 了 master_ name 表 。 


CREATE TABLE master_name ( 
id INT NOT NULL PRIMARY KEY AUTO INCREMENT, 
date added DATETIME, 
date_modified DATETIME, 
f name VARCHAR (75), 
1 _ name VARCHAR (75) 


fe PORK, FRG BSE, XERA Allmaster_ namek AK AK. 
例如 ，address 表 具有 上 月 已 的 主键 id 字段 以 及 date added 和 date modified 


段 ， 以 及 用 来 建立 关系 的 master_ id 字段 。 


master id 将 等 于 master name 表 中 的 id 字段 ， 匹 配 到 这 是 谁 的 地 址 。 

master id 字段 不 是 一 个 唯一 键 ， 因 为 一 个 人 拥有 多 个 地 址 条 目 绝对 是 合 
理 的 。 我 们 在 type 字 段 中 看 到 了 这 一 点 ， 这 个 字段 定义 为 包 侣 3 个 选项 
的 一 个 枚 举 类 型 ， 这 3 个 选项 是 home、work 或 other。 一 个 人 可 以 有 三 个 
类 型 中 的 一 个 或 多 个 ， 因 此 ， 除 了 主键 ia， 这 个 表 中 没有 其 他 的 键 。 假 
设 这 个 特定 的 地 址 短 只 包括 美国 的 地 址 ， 我 们 用 address、city、state 和 
zipcode Bx fa] WI THK « 
CREATE TABLE address { 

id INT NOT NULL PRIMARY KEY AUTO INCREMENT, 

master_id INT NOT NULL, 

date_added DATETIME, 

date modified DATETIME, 

address VARCHAR (255), 

city VARCHAR (30), 

State CHAR (2), 


Zipcode VARCHAR {19}, 
type ENUM ({'home', ‘work', Other ) 


telephone、fax 和 email 表 也 都 是 同一 主题 的 变形 ， 建 表 的 SQL 语句 
如 下 。 


CREATE TABLE telephone ( 
id INT NOT NULL PRIMARY KEY AUTO INCREMENT, 
master_id INT NOT NULL, 
date added DATETIME, 
date_modified DATETIME, 
tel number VARCHAR (25), 
type ENUM {'home', ‘work', Other ) 
) 
CREATE TABLE fax ( 
id INT NOT NULL PRIMARY KEY AUTO INCREMENT, 
master_id INT NOT NULL, 
date_added DATETIME, 
date modified DATETIME, 
fax number VARCHAR (25), 
type ENUM { home ， work ， ‘other' ) 
) 
CREATE TABLE email ( 
id INT NOT NULL PRIMARY KEY AUTO INCREMENT, 
master id INT NOT NULL, 
date_added DATETIME, 
date modified DATETIME, 
email VARCHAR (150), 
type ENUM ({‘home', ‘work', ‘other' ) 
)5 


personal_notes 表 也 这 从 同样 的 模式 ， 只 不 过 master_ id 是 一 个 唯一 键 


并 且 只 人 允许 每 人 一 条 notes 记 录 ，SQL 语 句 如 下 : 


CREATE TABLE personal_notes { 
id int NOT NULL PRIMARY KEY AUTO INCREMENT, 
master_id INT NOT NULL UNIQUE, 
date added DATETIME, 
date_modified DATETIME, 
note TEXT 


); 


既然 已 经 创建 了 表 ， 我 们 可 以 编写 用 来 管理 和 浏 
本 了。 


史记 录 的 表 蛙 和 脚 


20.2 Att Rae EE PALS FE 


在 第 19 半 中 ， 我 们 使 用 共同 函数 的 一 个 包含 文件 使 得 脚本 更 为 精 
简 。 在 本 章 中 也 是 这 样 。 尺 害 共 同 函 数 文 件 中 唯一 的 内 容 是 数据 库 连 接 
图 数 ， 这 个 过 程 还 是 能 起 到 两 个 作用 : 让 脚本 更 为 精 傈 ， 并 且 当 数据 库 
连接 的 信息 肥 生 变化 的 时 候 ， 不 用 在 多 个 文件 中 修改 这 一 信息 。 程 序 清 
单 20.1 包 含 了 本 革 中 的 脚本 共有 至 的 代 公 。 


程序 清单 20.1 包含 文件 中 的 共同 函数 


1: <?php 

2: function doDB() { 

da global $mysqli; 

4: 

55 //connect to server and select database; you may need it 
6: $mysqli = mysqli _connect("localhost", "“joeuser", 

T: "somepass’, ‘"testDB"); 

B: 

9: //1if connection fails, stop script execution 

10: if (mysqli connect_errno(})) { 

1 42 printf("Connect failed: %s\n", mysqli _connect_error{)); 
tee exit(); 

jee } 

14: } 

15s T 


第 2 行 到 第 14 行 建立 了 第 一 个 函数 doDBO， 这 征 一 个 简单 的 数据 库 
连接 函数 。 如 末 没 有 成 功 建立 连接 ， 在 调用 这 个 图 数 的 时 候 ， 脚 本 将 会 
退出 ;人寿 则 ， 它 将 让 $mysqli 的 值 可 供 脚 本 其 他 部 分 使 用 。 


把 这 个 文件 保存 为 ch20_include.php 并 将 其 放置 到 Web 服 务 器 上。 本 
革 中 的 其 他 代码 将 会 在 脚本 的 前 几 行 中 包含 这 个 文件 。 


20.3 BI #E—P SEE 


fe Ze HD ECR BL BO BRE, TALC, Oy BEB EPS SEL N 
的 。 程 序 清单 20.2 为 我 们 将 在 本 草编 写 的 所 有 脚本 创建 了 一 个 简单 的 
琳 单 ， 岂 做 mymenu.html。 


程序 清单 20.2 ”地址 籍 菜 E 


<!DOCTYPE html> 

<html> 

<head> 

<title>My Address Book</title> 
</head> 

<body> 

<hi>My Address Book</h1> 


o DO 上 oh 一 


9: <p><strong>Management</strong></p> 

10: <ul> 

11: <li><a href="addentry.php">Add an Entry</a></li> 
12: <li><a href="delentry.php">Delete an Entry</a></1i> 
13: </ul> 


15: <p><strong>Viewing</strong></p> 

16: <ul> 

17: <li><a href="selentry.php">Select a Record</a></1i> 
18: </ul> 

19: </body> 

20: </html> 


图 20-1 显 示 了 程序 消 单 20.2 的 输出 。 我 们 将 控 照 顺序 处 理 这 些 项 中 
的 每 一 个 ， 从 下 一 市 的 “Add an Entry” 开 始 。 


f EY My Address Book 


€ C | O http://localhost/20/mymenu.htm| 
My Address Book 


Management 


è Add an En 
® Delete an Entry 


Viewing 


图 20-1 ”地 址 夭 菜 E 








20.4 创建 记录 添加 机 制 


只 和 是 因为 我 们 要 淤 在 地 添加 信息 到 6 个 不 同 的 表 中 ， 并 不 意味 看 表 
单 或 脚本 会 很 庞大 。 实 际 上 ， 脚 本 看 上 去 和 在 前 面 的 章节 所 创建 的 脚本 
没有 太 大 差异 ， 并 且 稍 加 练习 ， 我 们 残 能 够 让 这 些 见 长 的 脚本 更 加 流畅 
Fill fey BL o 


在 程序 清单 20.3 中 ， 你 可 以 看 到 一 个 名 为 addentry.php 的 基本 记录 湛 
加 脚本 ， 它 有 两 部 分 : 如 有 果 要 显示 表单 应 该 做 些 什 么 (第 4 行 到 第 89 
行 ) 以 及 如 果 要 提交 表单 应 该 采 取 什 么 动作 第 91 行 到 第 187 行 )。 第 3 
行 到 第 89 行 只 是 简单 地 把 HTML 表 单 的 内 容 放 入 到 一 个 名 为 
$display_block 的 字符 串 中 。 


程序 清单 20.3 ”名 为 addentry.php 的 基本 记录 添加 脚本 


<?php 
include ‘ch20 include.php'; 


if {!$ POST) { 


ON Doh wd Dh + 


tO 


/ ¿haven't seen the form, so show it 

$display block = <<<END OF TEXT 

<form method="post" action="$ SERVER[PHP SELF]"> 

<fieldset> 

<legend>First/Last Names:</legend><br/> 

<input type="text" name="T_name" size="30" 
maxlength="75" required="required" /> 

<input type="text" name="1 name" size="30" 
maxlength="75" required="required" /> 

</fieldset> 


<p><label for="address">Street Address:</label><br/> 
<input type="text" id="address" name="address" 
$ize="30" {></p> 


<fieldset> 

<legend>City/State/Zip:</legend><br/> 

<input type="text" name="city" size="30" maxlength="5@" /> 
<input type="text" name= State size="5" maxlength="2" /> 

<input type="text" name="Zipcode" size="1@" maxlength="10" 
</fieldset> 


<fieldset> 

<legend>Address Type:</legend><br/> 

<input type="radio" id="add type_h" name="add type" 
value="home" checked /> 
<label for="add_type_h'>home</label> 

<input type="radio" id="add type w" name="add type" 
Value="work" /> 
<label for="add type w'>work</label> 

<input type="radio" id="add type o" name="add type" 
value="other" /> 
<label for= add type_o'>other</label> 

</fieldset> 


{> 


39; 
40: 
4l: 
42: 
43: 
44: 
45: 
46: 
47: 
48: 
49: 
50: 
-= ie 
D2: 
53: 
54; 
5b: 
56: 
5T: 
58: 
59: 
60: 
61: 
62: 
63: 
64: 
65: 
66: 
6r.: 
68: 
69: 
TÔ: 
Ti: 
T2: 
ret = 
74: 
fa: 
76: 
TE: 
r68: 
79: 
8: 
81: 
82: 
83: 
日 4 : 
85: 
86: 
87: 
88: 
89: 


<fieldset> 
<legend>Telephone Number:</legend><br/> 


<input type="text" name="tel number" size="30" maxlength="25" 


<input type="radio" 1d= tel type n° name="tel type" 
Value= home checked /> 
<label for="tel_type_h">home</label> 
<input type="radio" id="tel type w" name="tel type" 
Value="work' /> 
<label for="tel type w">work</label> 
<input type="radio" 1d= tel type D name="tel type" 
value="other" /> 
<label for="tel type o”">other</label> 
</fieldset> 


<fieldset> 
<legend>Fax Number:</legend><br/> 


<input type="text" name= fax number" size="30" maxlength="25" 


<input type="radio" id="Tax type h" name="Tax_ type" 
Value= honme checked /> 
<label for="fax type h">home</label> 
<input type="radio" id="fax type w" name="Tax type” 
value= work /> 
<label for="Tax_type_w">work</label> 
<input type="radio" id="fax_type_o" name="Tax_type'" 
Value="other" /> 
<label for= fax type o">other</label> 
</fieldset> 


<fieldset> 
<legend>Email Address:</legend><br/> 


<input type="email" name="email" size="3@" maxlength="150" 


<input type="radio" id="email type h" name="email type" 
value="home" checked /> 
<label for="email_ type_h">home</label> 
<input type="radio" id="email type_w" name="email type" 
Value= work /> 
<label for="email type w">work</label> 
<input type="radio" id="email type o" name="email type" 
Value="other" /> 
<label for="email type o*>other</label> 
</fieldset> 


<p><label for="note">Personal Note:</label><br/> 
<textarea id="note" name="note" cols="35" 
rows="3"></textarea></p> 


<button type="submit" name="submit" 
Value="send">Add Entry</button> 
</form> 


END OF TEXT; 


{> 


{> 


{> 


在 这 里 ， 先 稍 等 片刻 ， 确 保 你 知道 在 上 述 程序 清单 中 发 生 了 什么 。 
在 浏览 其 他 代码 之 前 ， 注 意 用 户 定 义 函 数 的 文件 在 第 2 行 被 包含 了 进 
来 。 正 如 已 经 提 到 的 ， 这 个 脚本 在 任何 时 间 都 执行 两 个 任务 中 的 一 个 : 
要 么 显示 添加 记录 表单 ， 要 么 执行 和 添加 一 条 新 记录 相关 的 SQL 查询 。 


确定 任务 的 逻辑 从 第 4 行 开 始 ， 通 过 测试 $_POST 的 仁 。 如 末 
$_POST 超 全 局 变量 中 没有 值 ， 用 万 必须 提交 表单 ， 并 且 因 此 需要 看 到 
表单 。 在 第 6 行 到 第 89 行 ， 表 单 的 HIML 放 入 到 了 一 个 名 为 
$display_block 的 字符 串 中 。 随 后 这 个 脚本 跳出 了 if...else 结 构 并 且 跳 转 
到 第 189 行 ， 在 那里 输出 HTML 并 且 显 示 出 $display_block 的 值 ， 在 这 个 
例子 中 就 是 该 表单 。 图 20-2 显 示 了 其 输出 。 


/ © Addan Enty x Ii: ë A O ë N 
€ C | © http://localhost/20/addentry.php ie] nx q 


Add an Entry 
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Street Address: 


| | 


Ciby State Zip: 
Doo MO O 





图 20-2 ”添加 记录 表单 


下 面 看 看 列表 中 市 有 让 语句 的 代码 ， 或 者 看 看 ， 如 条 表单 所 区 了 的 
话 ， 会 肥 生 毕 什 么 ， 如 程序 清单 20.3《〈 续 ) 所 示 。 


程序 清单 20.3 ”名 为 addentry.php 的 基本 记录 添加 脚本 《〈 续 ) 


} else if {$ POST) { 
//time to add to tables, so check for required fields 


Le (SPOSTAT Name | == =S) | Ce POst (<1 nate? | == 5]; 4 
header("Location: addentry.php") ; 
exit; 

I 


g7: {iconnect to database 


98: doDB(); 

99: 

100: {i create clean versions of input strings 

101: $safe f name = mysgli real escape string({$mysqli, 

102: $ POST['f_name']); 

103: $safe 1 name = mysqli real escape string($mysqli, 

104: $ PCST['1_name']); 

105: $safe address = mysqli real escape string(Smysqli, 

106: § POST['address']); 

107: S$safe city = mysqli real escape string(S$mysqli, 

108: $ POST['city']); 

1@9: Ssafe_state = mysqli_real_escape_string({Smysqli, 

110: $ POST['state']); 

111: $safe zipcode = mysqli _real_escape_string(Smysqli, 

112: § POST['zipcode']); 

113: $safe tel number = mysqli_real_escape_string({$mysqli, 
114: $ POST['tel_number']}; 

115: Ssafe fax number = mysqli real escape string({Smysqli, 
116: $ POST['fax_number']}; 

117: Ssafe email = mysqli_real_escape_string{S$mysqli, 

118: $ POST[ 'email']); 

119: $safe note = mysqli_real_escape string($mysqli, 

120: $ POST['note']); 

121: 

122: f/fadd to master_name table 

123: $add master sql = "INSERT INTO master_name (date_added, 
124: date modified, f name, 1 name) VALUES 

125: (now{), now{), '".$safe_f_name."', '".$safe_1l_name."')"; 
126: $add master_res = mysqli_query($mysqli, $add master sql) 
127! or die(mysqli error($mysqli}); 

128: 

129: f/fget master_id for use with other tables 

130: $master_id = mysqli_insert_id($mysqli); 

atl 

132: if {($_POST['address']) || (S$ _POST['city']) || 

133: ($ POST['state']) || (% POST['zipcode'])) { 

134: {i something relevant, so add to address table 

135: padd_address_sql = "INSERT INTO address (master_id, 
136: date added, date modified, address, city, state, 
137: Zipcode, type) VALUES 

138: ('".$master_id."', now(}, now(}, 

139: '" $safe_address."', '".$safe_city."', 

140: '".$safe_state."' , '".Ssafe_zipcode."' , 

141: "$$ POST['add type’']."'})"; 

142: $add address res = mysqli query(Smysqli, $add address sql) 
143: or die({mysoqli_error($mysqli)); 

144: } 

145: 

146: if {$_POST['tel number']) { 

147: /isomething relevant, so add to telephone table 

148: $add_tel_sql = "INSERT INTO telephone (master_id, date added, 
149: date modified, tel number, type) VALUES 

150: ('".$master id."', now(}, now(}, 

151: '" $safe tel number."', '".$ POST['tel type'].""}"; 
152: $add tel res = mysqli_query($mysqli, $add tel sdql) 
153: or die{mysoli error($mysqli)); 


154: } 
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160: 
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166; 
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188: 
189: 
199: 
191: 
192: 
193: 
194: 
195: 
196: 
197: 


if ($_POST['fax_number']) { 
/fsomething relevant, so add to fax table 
$add fax sql = "INSERT INTO fax {master_id, date added, 
date modified, fax_number, type) VALUES 
('".$master_ id."', now(), now({), '".$safe fax number."', 
'" $ POST['fax_type']."'}" 
$add_fax_res = mysqli querytgmysdii, $add_fax_sql) 
or die{mysqli error($mysqli)); 
} 
if ($ POST['email'])} { 
//something relevant, so add to email table 
$add email sql = "INSERT INTO email (master id, date added, 
date _ modified, email, type} VALUES 
('".$master_id."', now(), now(), ‘".$safe_email."', 
“maS POST['email type']."')"; 
$add email res = mysqli query($mysqli, $add email sql} 
or die{mysqli error($mysqli) ); 


} 


if ($ POST['note']) { 
{//something relevant, so add to notes table 
$add notes sql = "INSERT INTO personal notes (master id, 
date added, date modified, note} VALUES 
('".$master_id."', now(}, now(), 
'". $safe note."')"; 
$add notes res = mysqli _query($mysqli, $add notes sql} 
or die({mysqli error($mysqli) ) ; 
} 
mysqli close($mysql1i}; 
$display block = "<p>Your entry has been added. Would you 
like to <a href=\"addentry.php\">add another</a>?</p>"; 
} 
?> 
<!DOCTYPE html> 
<head> 
<title>Add an Entry</title> 
</head> 
<body> 
<h1>Add an Entry</h1> 
<?php echo $display block; ?> 
</body> 
</html> 


如 果 $_POST 中 有 值 的 话 ， 调 用 第 90 行 的 else 条 件 ， 这 意味 着 用 户 已 


经 捉 区 了 和 表单。 在 这 个 简单 的 例子 中 ， 有 两 个 字段 被 设 置 为 必需 字段 ， 


即 人 物 的 名 和 姓 。 因 此 ， 第 92 行 到 第 95 行 检查 $_POST [Tf_name'] 和 
$_POST [1_name'] 中 的 值 ， 并 且 如 果 任 何 一 个 值 不 存在 的 话 就 把 用 户 重 
定 同 回 到 表单 。 


提示 : 


由 于 表单 中 的 这 两 个 必需 字段 标记 为 使 用 HTML 5 的 required 必 性， 如 果 这 两 个 字段 中 没 
有 文本 的 话 ， 表 单 将 不 会 提交 。 然 而 ， 对 于 较 早 的 浏览 器 ， 或 者 那些 不 支持 INPUT 字 段 中 的 
HTML 5 的 required 属 性 的 设备 ， 这 一 服务 器 病 检测 不 会 触发 。 





在 检 租 过 必需 的 字段 之 后 ， 我 们 在 第 98 行 连接 到 数据 库 。 一 旦 连接 
了 数据 库 ， 可 以 安全 地 使 用 mysqli real _escape_string0 来 检测 用 户 输 
和 入， 并且 创建 这 些 字 符 吕 的“ 安全? 版本， 以 供 INSERT 语 句 使 用 。 


接 下 来 是 多 条 插入 语句 ， 其 中 只 有 一 条 是 必需 的 ， 允 是 加 
master_name 插 入 一 条 记录 。 这 发 生 在 第 123 行 到 第 127 行 。 在 插入 之 
后 ， 在 第 130 行 使 用 mysqli insert_id0 把 这 条 记录 的 id 提取 了 出 来 。 我 们 
在 其 余 的 SQL 奏 询 中 使 用 了 这 个 什 ， 它 现在 家 引用 为 Smaster_ id。 


把 记录 插入 到 其 他 表 中 的 SQL 得 询 都 是 有 条 件 的 ， 这 意味 痢 只 有 当 
菏 个 条 件 为 true 的 时 候 ， 它 们 才能 发 生 。 在 第 132 行 到 第 133 行 ， 我 们 看 
到 必须 满足 的 条 件 是 ， 下 列 变 量 中 的 任何 一 个 全 存在 : $_POST 
[address]、$_POST ['city']. $_POST [state] 和 $_POST ['zipcode’]. WR 
条 件 满足 ， 第 135 行 到 第 143 行 创建 并 执行 一 个 查询 。 


对 于 癌 telephone 表 〈 第 146 一 154 行 ) 、fax 表 《第 156 一 164 行 ) 、 
email (%175~183í7) 以 及 personal_notes 表 〈 第 123 一 130 行 ) “SII 
内 容 ， 同 样 的 情况 也 成 这 。 如 果 条 件 满 足 ， 记 录 将 会 插入 到 这 些 表 中 。 


一 旦 通过 了 这 组 条 件 ， 给 用 户 的 消息 也 放置 到 $display_block 变 量 
中 ， 脚 本 退出 这 个 if...else 结 构 并 且 在 第 189 一 197 行 显示 HITML。 图 20-3 
展示 了 记录 添加 脚本 的 一 个 输出 。 


© Add an Entry 


= Cie http://localhost/20/addentry.php Ki KE‘ 


Add an Entry 


Your entry has been added. Would you like to add another? 


图 20-3 EZAM S -ZUK 


1 Ik PS eS I LAR, MERI NA EE EP 
中 使 用 。 目 己 签 试 修改 这 个 脚本 ， 以 便 在 成 功 地 插入 记录 后 ， 把 表 早 输 
EL AN EE BEE 


20.5 浏览 记录 


如 末 要 通过 MySQL 监 视 喜 或 者 其 他 的 寞 面 来 执行 咎 询 以 验证 在 上 
一 节 中 的 工作 ， 我 们 可 能 会 对 每 个 表 都 要 输入 SELECT * FROM... 感 到 
厌烦。 在 本 中 ， 我 们 将 创建 一 个 由 两 部 分 组 成 的 脚本 ， 癌 你 展示 如 何 
从 数据 库 选 取 和 浏 虎 记录 。 


程序 清单 20.4 展 示 了 名 为 selentry.php 的 选择 和 浏览 的 脚本 ， 它 拥有 3 
部 分 : 记录 选择 表单 (第 5 行 到 第 42 行 ) 和 显示 记录 内 容 的 代码 (第 43 
行 到 第 172 行 ) ， 以 及 显示 动态 产生 的 字符 串 的 HTML 模板 〈 第 176 行 到 
党 184 行 )。 由 于 这 段 代 码 很 长 ， 我 们 将 其 分 解 为 较 小 的 代码 段 来 讨 


it. 


程序 清单 20.4 用 来 选取 和 浏览 记录 的 名 为 selentry.php 的 脚本 


1: <?php 

2: include ‘ch2@ include.php' ; 

3: doDB() 

4: 

a: If (1S. POST) yf 

6: /fhaven't seen the selection form, so show it 

73 $display_ block = "<hi>Select an Entry</hi>"; 

8: 

9: / ¿get parts of records 

10: $get_list_sql = "SELECT id, 

TIR CONCAT_WS(', ', 1 name, f_name} AS display_name 
T2: FROM master name ORDER BY 1 name, f name"; 

13% $get_list_res = mysqli_query($mysqli, $get_list_sql) 

14: or die(mysqli error($mysqli)); 

153 

16: if (mysqli num rows($get_ list res) < 1) { 

Fi- //no records 

Le: $display block .= "<p><em>Sorry, no records to select!</em></p>"; 
19: 

20: } else { 

21: iihas records, so get results and print in a form 

225 $display block .= " 

23: <form method=\"post\" action=\"".$_SERVER['PHP_SELF']."\"> 
24: <p><label for=\"sel id\">Select a Record:</label><br/> 

25: <select id="sel_ id\" name=\"sel_id\" required=\"required\"> 
26: <option value=\"\">-- Select One --</option>' ; 

27 : 

28: while ($recs = mysqli fetch array($get list res)) { 

29: $id = $recs['id']; 

30: $display name = stripslashes($recs['display name']); 
i ie $display block .= 

32: "<option value=\"".$id."\">".$display name. "</option>"; 
33: } 

34; 

35: $display block .= " 

36: </select> 

37: <button type=\"submit\" name=} "submit" 

38: Value=\"view\">View Selected Entry\"></button> 
39: </form>"; 

40: } 

41: :ifree result 

42: mysqli free result({$get list res); 


和 addentry.php 脚 本 一 样 ，selentry.php 脚 本 在 任何 时 候 都 执行 两 项 任 
务 中 的 一 个 : 要 么 显示 选择 表单 ， 要 么 执行 与 浏览 记录 相关 的 所 有 SQL 
答 询 。 不 管 执 行 两 个 任务 中 的 哪 一 个 ， 都 要 用 到 数据 库 。 因 此 ， 我 们 在 


第 2 行 导 入 了 市 有 连接 函数 的 文件 ， 并 且 在 第 3 行 调用 该 函数 。 


确定 任务 的 逻辑 在 第 5 行 开 始 ， 通 过 测试 超 全 局 变量 $_ POST 的 值 来 
实现 。 如 条 $_POST 没 有 值 ， 表 示 用 户 还 没有 看 到 选择 表单 ， 因 此 需要 
看 到 它 。 在 第 7 行 开 始 一 个 名 为 $display_block 的 字符 串 ， 并 且 这 个 字符 
串 最 终 将 存储 组 成 记录 选取 表单 的 HTML 。 


在 第 10 行 到 第 12 行 ， 我 们 从 master_name 表 的 记录 中 选择 了 特定 的 
字段 来 构建 表单 中 的 选择 下 拉 选 项 。 从 这 一 步 开 始 ， 我 们 只 需要 想 要 选 
择 的 记录 的 人 的 名 字 和 ID。 第 16 行 测试 了 得 询 的 结果 ， 如 采 租 询 没有 续 
果 ， 我 们 不 会 构建 一 个 表 蛙 。 如 果 是 这 种 情况 ，$display_block 的 值 将 会 
由 一 条 错误 消息 填 宛 ， 并 且 脚 本 将 会 结束 ， 把 最 终 的 HTIML 时 示 到 屏 用 
Mise 


然而 ， 让 我 们 假设 在 master_name 表 中 有 几 条 记录 。 在 这 个 例子 
中 ， 我 们 必须 从 查询 结果 中 提取 信息 ， 以 便 能 够 构成 表单 。 这 在 第 28 行 
到 第 33 行 完成 ， 写 入 到 $display_block 字 符 串 的 表单 元 素 都 在 这 些 行 中 。 


我 们 在 第 42 行 停止 了 这 个 列表 ， 但 是 ， 很 快 将 看 到 第 43 行 到 这 个 脚 
本 结束 的 内 容 。 如 果 我 们 想 要 关闭 这 个 计 语 句 以 及 PHP 代 码 块 ， 并 且 在 
此 时 把 $display_block 的 值 显 示 到 屏 巢 上 ， 将 会 看 到 如 图 20-4 所 示 的 结 

(可 能 具有 不 同 的 数据 条 目 ) 。 


| My Records i 
€ 3 C |O http://localhost/20/selentry.php Ki KE w 


Select an Entry 


Select a Record 
Doe, Jane |x| 
-- Select One 一 
Able, Annie 
Doe, Jane 


smith, John 







图 20-4 ”记录 选择 表单 


然而 ， 我 们 必须 完成 selentry.php 脚 本 ， 因 此 ， 我 们 从 第 43 行 开始 继 
续 程 序 清单 20.4， 从 这 里 开始 了 if...else 语 名 的 else 部 分 ， 如 下 所 示 。 


程序 清单 20.4 用 来 选取 和 浏览 记录 的 名 为 selentry.php 的 脚本 《〈 续 ) 


} else if ($_POST) { 
¿icheck For required fields 


if be POSTI” sel 0] ea 87 yf 
header("Location: selentry.php"); 
exit; 

} 


cpPreate safe version of ID 
$safe id = mysqli real escape string($mysqli, $ POST['sel id']);52: 


/iget master info 

$get_master_sql = "SELECT concat_ws{' ',f_name,l_ name) as display_name 
FROM master_name WHERE id = ‘".$safe_id."'"; 

$get master res = mysqli query($mysqli, $get_ master sql} 
or die(mysqli_error($mysqli))}; 


while {$name info = mysqli fetch _array($get_master_res}} { 
$display name = stripslashes($name_info[ 'display_name' ]}; 
} 


$display_block = "<h1>Showing Record for ".$display_name."</h1i>"; 


free result 
mysqli free _result($get_master_res} ; 


/iget all addresses 
$get addresses sgl = "SELECT address, city, state, zipcode, type FROM 
address WHERE master_id = '".$safe_id."'"; 
$get_addresses res = mysali gquery($mysoqli, $get addresses sgl) 
or die(mysqli_error($mysq1li) ) ; 


if (mysqli_num_rows($get_addresses res) > @) { 
A e DL .= "<p><strong>Addresses:</strong><br/> 
<u aS 


while ($add info = mysqli fetch array($get addresses res)) 1 
address = stripslashes($add_info[ ‘address’ ]}; 
$city = stripslashes($add info['city']}; 
$state = stripslashes($add_ info['state']}; 
$Zipcode = stripslashes($add info[ 'zipcode']); 
$address type = $add_info['type']; 


$display block .= "<li>$address $city $state $zipcode 
($address type)</li>': 
} 
$display block .= "</ul>"; 


} 
/iftree result 
mysqli free result(Sget_addresses res); 


第 43 行 包含 了 证 ...else 语 句 的 else 部 分 ， 并 且 如 采 表 单 想 要 看 一 条 特 
定 的 记录 束 调 用 这 一 部 分 。 我 们 首先 在 第 45 行 租 看 一 个 所 需 的 字段 ， 在 
这 个 例子 中 束 是 $_POST ['sel_id"] 的 值 。 这 个 值 把 来 自 master_name 表 的 
ID 和 和 在 记录 选择 表单 中 做 出 的 选择 的 ID 进 行 比较 。 如 果 选 择 的 ID 不 存 
在 ， 用 户 将 重 定 同 到 选择 表单 ， 当 主键 不 存在 的 时 候 ， 我 们 无 法 很 好 地 
从 一 组 表 中 收集 信息 。 


假设 有 一 个 值 赋 给 了 $_POST[“sel_id”]， 在 第 51 行 创建 了 它 的 一 个 
安全 版 本 。 接 下 来 ， 我 们 在 第 54 行 到 第 57 行 执行 一 个 查询 ， 它 获取 了 我 
们 想 要 浏览 的 记录 的 用 户 的 名 字 。 这 个 信息 放置 在 我 们 现在 已 经 熟悉 的 
$display_block 和 字符 串 中 ， 随 看 脚本 继续 ， 这 个 字 从 串 将 继续 构建 。 


第 69 行 到 第 89 行 表示 根据 address 表 的 一 个 查询 结果 构建 最 终 的 显 
示 。 如 采 所 选择 的 个 人 在 address 表 中 没有 记录 ， 那 下 不 会 有 什么 内 容 添 
加 到 $display_block 字 符 串 。 然 而 ， 如 果 有 一 个 或 多 个 条 目 ， 这 个 人 的 地 
址 将 会 壕 加 到 $display_block 字 人 符 串 作为 一 个 或 多 个 未 排序 的 列表 元 勾 ， 
如 第 78 行 到 第 87 行 所 示 。 


程序 清单 20.4 的 第 92 行 到 第 168 行 执行 相同 类 型 的 循环 并 且 写 入 到 
$display_block 变 量 ， 但 是 ， 操 作 的 表 不 同 。 例 如 ， 第 92 行 到 第 109 行 得 
看 telephone 表 中 的 信息 ， 并 且 创 建 相应 的 字符 串 以 供 添 加 到 
$display_block， 如 果 存 在 任何 信息 的 话 。 同 样 的 结构 在 第 114 行 到 第 130 
行 针 对 fax 表 的 信息 而 重复 ， 在 第 135 行 到 第 150 行 针对 email 表 的 信息 而 
重复 ， 在 第 155 行 到 第 172 行 针对 personal_notes 表 中 出 现 的 信息 而 重复 。 


程序 清单 20.4 用 来 选取 和 浏览 记录 的 名 为 selentry.php 的 脚本 《〈 续 ) 


92 

93: 
94; 
95: 
96: 
aft 
98; 
99: 


1090: 
10i: 
102: 
103: 
104: 
105: 
106: 
107: 
108: 
109: 
110: 
11i: 
112: 
113: 
114: 
115: 
116: 
117: 
118: 
119: 
120: 
121; 
122: 
123: 
124: 
125: 
126: 
127: 
128: 
129: 
130: 
131: 
132: 
133: 
134: 
135: 
136: 
Tait 
138: 
139: 
140: 
141: 
142: 
143: 
144: 
145: 
146: 
147: 


92: f/fget all tel 92220 


$get tel sql = "SELECT tel number, type FROM telephone WHERE 
master_id = '".$safe_id."'"; 
$get tel res = mysqli query({Smysqli, $get_tel sql) 
or die(mysqli_error(Smysqli)); 


if (mysqli_num_rows(S$get_tel_res) > @) 1 
$display block .= "<p><strong>Telephone: </strong><br/> 
<ul>" i 


while ($tel_info = mysgqli_fetch_array($get_tel_res}} { 
$tel number = stripslashes($tel info['tel number']); 
$tel type = $tel info['type']; 


$display_block .= "<li>$tel_number ($tel_type)</1li>"; 
} 
$display_ block .= "</ul>"; 
} 
/ffree result 
mysqli_free_result($get_tel_res) ; 


/ iget all fax 
$get fax sql = "SELECT fax number, type FROM fax WHERE 
master_ic = "“.$safe_id."'"; 
$get_fax_res = mysqli_query{$mysqli, $gqet_fax_sql) 
or die(mysqli_ error($mysqli)); 


if (mysqli_num rows($get_fax_res) > Ø) { 
$display block .= "<p><strong>Fax:</strong><br/> 
<ul>"; 


while ($fax_info = mysqli_fetch_array($get_fax_res)) { 
$fax number = stripslashes($fax_info['fax_number' ]}; 
$fax type = $fax_info['type']; 


$display_block .= "<li>$fax_number ($fax_type)</1i>"; 
} 
$display block .= "</ul>"; 
} 
//free result 
mysoli free result($get_ fax res); 


/fget all email 
$get_email sql = "SELECT email, type FROM email WHERE 
master_id = '".$saTfe_id."'"; 
$get_email_res = mysqli_query($Smysqli, Sget_email sql) 
or die(mysqli_ error($mysqli)); 
if (mysqli _num_rows($get_email res} > @) { 
$display block .= "<p><strong>Email:</strong><br/> 
<ul>"; 


while ($email_info = mysqli_fetch_array($get_email_res}) { 


$email = stripslashes($email info['email']); 
$email type = $email_info['type']; 


$display block .= "<li>$email ($email type)</1li>"; 


148: } 


149; $display block .= "</ul>"; 

159: } 

151: //free result 

152: mysqli free result($get_email res); 

153: 

154: /fget personal note 

155: $get notes sol = "SELECT note FROM personal notes WHERE 

156: master_id = '".$safe_id."'"; 

Lor: $get notes res = mysqli query(Smysqli, $get notes sql} 

158: or die(mysqli_error($mysqli)); 

159: 

160: if (mysqli num rows($get notes res) == 1) 1 

161: while ($note_info = mysqli _fetch_array{$get_notes res)) 1 
162: Gnote = nl2br(stripslashes($note info[ 'note']}}; 

163: } 

164: $display block .= "<p><strong>Personal Notes:</strong><br/> 
165: Fnote</p>"; 

166: } 

167: //free result 

168: mysqli free result($get notes res); 


我 们 必须 对 脚本 做 一 些 清理 和 结束 工作 ， 如 程序 清单 20.4 的 最 后 一 
部 分 所 示 。 


程序 清单 20.4 用 来 选取 和 浏览 记录 的 名 为 selentry.php 的 脚本 “〈 续 ) 


169: $display_block .= "<br/> 

170: <p style=\"text-align:center\ > 

1713 <a href=\"".$ SERVER['PHP SELF']."\">select another</a></p>"; 
Aves 下 


173: //close connection to MySQL 
174: mysqli _close($mysqli) ; 

175: T> 

176: <!DOCTYPE html> 

177: <html> 

178: <head> 

179: <title>My Records</title> 
180: </head> 

181: <body> 

182: <?php echo $display block; ?> 
183: </body> 

184: </html> 


在 第 172 行 结束 让 ...else， 以 及 稍 后 结束 PHP 代 码 块 之 前 ， 我 们 在 第 
169 行 到 第 171 行 ， 简 单 地 显示 出 回 到 选择 表单 的 一 个 链接 。 第 176 行 到 
脚本 结束 都 是 我 们 用 来 包含 $display_block 字 符 串 的 内 容 的 HIML 通 用 模 
板 。 


在 从 图 20-4 所 示 的 表单 选择 一 条 记录 之 后 ， 我 们 将 会 看 到 如 图 20-5 
所 示 的 结果 ， 当 然 ， 你 的 数据 会 有 所 不 同 。 


= | ©) |e Se 


a) My Records 


€ C | O http://localhost/20/selentry.php Kin « 


Showing Record for Jane Doe 


Addresses: 
èe 123 Main Street Washington DC 20011 (home) 


Telephone: 


mM 


e 555-555-1212 (home) 


Fax: 


Email: 
* jane(@doe.com (home) 


Personal Notes: 
Jane is pretty cool. 


图 20-5 一 条 个 人 记录 


当 根 据 目 己 的 记录 目 行 妾 试 这 个 脚本 的 时 候 ， 只 有 和 针对 那些 拥有 和 
他 们 相关 联 的 附加 数据 的 个 体 ， 才 会 看 到 信息 。 


例如 ， 如 果 你 的 一 个 朋友 有 一 个 条 目 ， 并 且 你 所 拥有 的 只 是 在 
email 表 输入 的 一 个 Email 地 址 ， 我 们 不 会 看 到 和 地 址 、 电 话 、 传 真 或 个 
人 记录 相关 的 其 他 任何 文字 ， 因 为 没有 在 这 些 表 中 输入 相关 的 数据 。 


20.6 ”创建 记录 的 删除 机 制 


记录 删除 机 制 事实 上 和 用 来 浏览 一 条 记录 的 脚本 相同 。 实 际 上 ， 我 
们 只 要 选取 程序 清单 20.4 中 的 前 42 行 ， 将 它们 粘贴 到 一 个 名 为 
delentry.php 的 新 文件 中 ， 然 后 在 第 24 行 和 第 38 行 把 “View” 修 改 
为 “Delete” 束 可 以 了 。 
从 第 43 行 开始 ，delentry.php 的 代码 的 剩余 部 分 如 程序 清单 20.5 所 
不 。 
程序 清单 20.5 用 来 选取 和 删除 记录 的 名 为 delentry.php 的 脚本 


43: } else if {$_POST} { 


44: //check for required fields 

45: if ($ POST['sel_id'] == "") { 

46 : header("Location: delentry.php"}; 
47: exit; 

48: } 

49 

50: //create safe version of ID 


513 $safe id = mysqli real escape string($mysqli, $ POST['sel id']}; 


53: {issue queries 


54: $del_master_sql = "DELETE FROM master_name WHERE 

55; id = *" S$safe 二 人 

5G. $del master res = mysqli query($mysqli, $del master sql) 
57: or die(mysqli error($mysqli) ) ; 

08: 

gg: $del address sql = "DELETE FROM address WHERE 

60 : id = '".$safe_id.."'"s 

61: $del address res = mysqli query($mysqli, $del address sql) 
62: or die(mysqli error($mysqli)); 

63: 

64: $del_tel_sql = "DELETE FROM telephone WHERE id = '".$safe_id."'"; 
65: $del_tel_ res = mysqli _query($mysqli, $del tel sql) 

66: or die(mysqli error($mysqli)); 

ar 

68: $del fax sql = "DELETE FROM fax WHERE id = '".$safe id."'"; 
69: $del fax res = mysqli _query($mysqli, $del_fax_sql) 

70: or die(mysqli error($mysqli)); 

Tin 

Tes $del email sql = "DELETE FROM email WHERE id = '".$safe id."'"; 
ta $del email res = mysqli _query($mysqli, $del email sql) 

74: or die(mysgqli_error($mysqli) ) ; 

roe 

76: $del_note_sql = "DELETE FROM personal_notes WHERE 

Ti: id = “t2nsarte aden Tin 

face $del note res = mysqli query({$mysqli, $del note sql) 

79: or die{mysqli error($mysqli) ) ; 

80: 

81: mysqli close($mysqli) ; 

02s 

83: $display_ block = "<hi>Record({s) Deleted</h1i> 

84: <p>Would you like to 

65: <a href=\"".$_SERVER['PHP_SELF']."\">delete another</a>?</p>"; 
86: } 

B74) > 

88: <!DOCTYPE html> 

89: <html> 

90: <head> 

91: <title>My Records</title> 

92: </head> 

93: <body> 

94: <?php echo $display block; ?> 

95: </body> 

96: </html> 


从 第 45 行 开始 ， 脚 本 碍 找 所 需 的 字段 $ POST ['sel_id'], MAET 
selentry.php 脚 本 中 所 做 的 那样 。 如 末 所 需 的 信 不 人 存在， 脚本 将 用 户 重 定 
和 器 到 选择 页 面 ， 但 是 ， 如 打 它 存在 ， 在 第 51 行 为 其 创建 一 个 安全 版 本 。 


在 第 54 行 到 第 79 行 ， 俘 询 从 所 有 的 表 中 删除 所 有 和 选择 的 个 体 相关 的 信 
恩 。 第 83 行 到 第 85 行 把 一 条 消息 放 入 $display_block 中 ， 脚 本 退出 的 时 候 
把 它 以 HTML 显 示 到 屏幕 。 图 20-6 显 示 了 记录 删除 脚本 的 一 个 输出 。 

(©) My Records 


= El | K | 
< 


C | © http://localhost/20/delentry.php 


Ki 也 S 


Record(s) Deleted 


Would you like to delete another? 


图 20-6 ”删除 一 条 记录 


当 删 除 一 条 记录 之 后 ， 返 回 到 记录 选择 页 面 ， 你 将 会 注意 到 删除 的 
个 体 不 再 存在 于 选择 页 面 中 ， 本 来 吏 该 如 此 ! 


20.7 “为 一 条 记录 这 加 子 条 目 


本 章 至 此 ， 我 们 已 经 学 习 了 如 何 添加 、 删 除 和 浏览 记录 。 还 缺 的 吏 
是 当 我 们 已 经 输入 了 一 条 记录 之 后 ， 同 相关 的 表 洪 加 附加 的 条 目 ， 例 
a. 我 们 只 需要 对 已 有 脚本 略 作 
改变 就 能 做 到 这 一 点 


在 程序 清单 20.4 的 selentr.php 脚 本 中 ， 把 第 185 行 到 第 186 行 修改 为 : 


$display block .= "<p style=\"text-align:center\"> 
<a href=\"addentry.php?master_id=".$ POST['sel_id']."\">add info</a> ... 
<a href=\"".$ SERVER[ PHP SELF']."\">select another</a></p>"; 


ØA A Æ [eladdentry.php hl A wasn SSH, FFA 
$ GET ['master_id'] 传 递 一 个 可 访问 的 变量 。 


现在 ， 为 了 达到 广 目 的 ， 我 们 需要 修改 程序 清单 20.3 中 的 
addentry.php 脚 本 。 下 面 是 对 最 初 的 脚本 的 修改 的 一 个 概括 。 


使 用 如 下 代码 段 奉 换 最 人 切 的 addentry.php 的 前 10 行 。 


<?php 
include “ch20 include.php ; 

doDB(}; 

if ((!1$ POST) || {$ GET['master id'] T= "")) { 


/fhaven't seen the form, so show it 

$display block = " 

<form method=\"post\" action=\"".$ SERVER['PHP_SELF']."\">"; 
if (isset($ GET['master id']}) { 


f/fcreate safe version of ID 
$safe_id = mysqli_real_escape string($mysqli, $ GET['master_id']); 


{iget first, last names for display/tests validity 
$get_names sql = "SELECT concat_ws(' ', f_name, 1 _ name) AS display name 
FROM master name WHERE id = '".$safe_id."'"; 
$get_names_ res = mysqli query($Smysqli, $get names sql} 
or die{mysqli error($mysqli)) ; 


if (mysqli num rows($get names res) == 1) { 
while ($name_info = mysqli _fetch_array($get_names res)) { 
$display name = stripslashes($name info['display name']); 


} 
} 
} 
if (isset($display name)) { 
$display block .= "<p>Adding information for 
<strong>$display_name</strong>:</p>"; 
} else { 
$display block .= <<<END OF_TEXT 
<fieldset> 


<legend>First/Last Names:</legend><br/> 
<input type="text" name="f_name" size="30" 
maxlength="75" required="required" /> 
<input type="text" name="1l_ name" size="30" 
maxlength="75" required="required" /> 
</fieldset> 
END OF TEXT; 


} 
$display block .= <<<END OF TEXT 


<p><label for="address">Street Address:</label><br/> 

APARER R EE S RANK RAS ENS -RIRKA 
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字 将 从 数据 库 中 提取 出 来 ， 为 了 美观 也 为 了 对 ID 进行 有 效 性 检查 。 


接 下 来 ， 在 最 礼 的 addentry.php 脚 本 中 找到 如 下 的 一 行 。 


<button type="submit’ name="submit" value="send">Add Entry</button> 


十 接 在 其 上 面 添加 如 下 内 容 。 


END OF TEXT; 


if ($ GET) { 
$display block .= "<input type=\"hidden\" name=\"master id\" 
Value=\ I I s $ GET[ ‘master 1d | ] ll ii MS l 
} 


$display block .= <<<END OF TEXT 
XE BU S master_idiN 212129 R—“MES, UREA. 
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到 表 中 ， 最 后 是 获取 $master_id 的 值 。 这 些 代 人 码 行 应 该 用 以 下 内 容 从 
“nee 


f//time to add to tables, so check for required fields 
if ((($ POST['T name'] == "") || {$ POST[ 1L name'] == "")} && 
{!isset({$ POST['master_id']})) 1 

header("Location: addentry.php"); 

exit; 


} 


:i:connect to database 
doDB(}; 


// create clean versions of input strings 

$safe f name = mysqli real _escape_string({$mysoqli, 
$ POST['f name']); 

$safe 1] name = mysqli_real_ escape string($mysqli, 
$ POST['l name']); 

$safe address = mysqli real escape string($mysqli, 
$ POST[ 'address']); 

safe city = mysqli real_escape string(Smysqli, 
$ POST['city']); 

$safe state = mysqli real escape string({$mysqli, 
$ POST[ 'state']); 

$safe zipcode = mysqli real escape string({$mysqli, 
$ POST[ zipcode’ ]); 

$safe tel number = mysqli real escape string(Smysqli, 
$ POST[ tel number']}; 

$safe fax number = mysqli real escape string($mysqli, 
$ POST[ ' fax_number' }}; 

$safe email = mysqli real escape string{$mysql1li, 
$ POST[ 'email']}; 

$safe note = mysqli real_escape string($mysqli, 
$ POST['note']); 


if {(!$ POST['master id']) { 
¿iadd to master name table 
fadd master sql = “INSERT INTO master name {date added, date modified, 
f name, l name) VALUES (now(), now(}, 
'" $safe f name."', '".$safe 1 name."')"; 
$add master res = mysqli query(Smysqli, $add master sql) 
or die{mysqli_ error({$mysqli)); 


¿iget master_id for use with other tables 

$master id = mysqli insert id{$mysqli); 
} else { 

$master id = mysqli real escape string(Smysqli, $ POST['master id']}; 
i 


这 些 代 人 码 修 改 了 对 所 需 字 上 段 的 检查 ， 人 允许 脚本 在 没有 姓 和 名 的 情况 
下 继续 ， 但 是 ， 只 有 当 它 有 一 个 $_ POST ['master_id'] 值 时 才 可 以 。 

然后 ， 脚 本 连接 到 数据 库 来 执行 我 们 和 希望 它 执行 的 所 有 添加 ， 但 
fe, URS _POST [master id] 的 值 已 经 存在 的 话 ， 它 会 略 过 对 


master namexeH 70. 


最 后 ， 在 脚本 的 处 理 插 入 到 personal_notes 表 的 SQL 语句 部 分 ， 把 
INSERT 改 为 UPDATE 来 处 理 notes 字 段 的 更 新 。 


$add notes sql = "UPDATE personal notes set note = '".$safe note."', 
date modified = now() WHERE master id = ‘'".$master id."'"; 


新 的 脚本 应 该 如 程序 清单 20.6 所 示 。 


程序 清单 20.6 ”新 的 addentry.php 脚 本 
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<?php 
include ‘ch2@ include.php' ; 
doDB{}; 


if ({!$ POST) || ($ GET['master_id'] != "")) { 
/ihaven't seen the form, so show it 
$display block = " 
<form method=\"post\" action=\"".$ SERVER['PHP_SELF']."\">"; 
if (isset($ GET['master id'])) { 
//create safe version of ID 
$safe id = mysqli real escape string($mysqli, $ GET[ ‘master id ] ) 


¿iget first, last names for display/tests validity 
$get names sql = "SELECT concat_ws(' ', f_name, 1 name) AS 
display name 
FROM master name WHERE id = '".$safe_id."'"; 
$get names res = mysqli query($mysqli, $get names sql) 
or die(mysqli_error($mysqli) ); 


if (mysqli num rows($get_ names res) == 1) { 
while ($name_info = mysqli fetch array($get names res}) { 
$display name = stripslashes($name info['display name']}; 


} 
} 
} 
if (isset($display_name)) { 
$display block .= "<p>Adding information for 
<strong>$display_ name</strong>:</p>"; 
} else 4 
$display block .= <<<END OF_TEXT <fieldset> 


<legend>First/Last Names:</legend><br/> 
<input type="text" name="f name" size="30" 
maxlength="75" required="required" /> 
<input type="text" name="1 name" size="30" 
maxlength="75" required="required" /> 
</fieldset> 
END OF_TEXT; 
} 
$display_block .= <<<END_OF_TEXT 
<o><label for="address'">Street Address:</label><br/> 
<input type="text" id="address" name="address" 
Size="30" {></p> 


<fieldset> 


45 : 
46: 
47: 
48 : 
49: 
50: 
51: 
Be: 
53: 
54: 
55: 
56: 
Or: 
58: 
59: 
60: 
61: 
62: 
63: 
64: 
65: 
66: 
Gi: 
68: 
69 : 
ro: 
Ti: 
Je: 
Ta: 
74: 
ro: 
T6: 
ff: 
r8: 
r9: 
80: 
61: 
82: 
83: 
64: 
85: 
86: 
os: 
&&: 
89: 
gg: 
91: 
92: 
93: 
94: 
95: 
96: 
gy: 
98: 
99; 
10@: 
101: 


<legend>City/State/Zip:</legend><br/> 

<input type="text" name="city" size="30" maxlength="50" /> 
<input type="text" name="state" size="5" maxlength="2" /> 
<input type="text" name="7iscode" size="10" maxlength="10" /> 
</fieldset> 


<fieldset>= 

<legend>Address Type:</ legend><br/> 

<input type="radio" id="add type hn name="add type" 
value="home" checked /> 
<label for="add type_h">home</label> 

<input type="radio" id="add type w name= add type” 
Value="work" /> 
<label for="add type w">work</label> 

<input type="radio" id="add type D name= adgdq type" 
Value="other" /> 
<label for="add type _o">other</label> 

</fieldset> 


<fi1eldset> 


<legend>Telephone Number:</legend>=<br/> 
<input type="text" name="tel number" size="30" maxlength="25" 
<input type="radio" id="tel_type_h" name="tel type" 
Value="home" checked /i> 
«label for="tel_ type_h">home</label> 
<input type="radio" id="tel type w" name="tel type" 
Value="work" /> 
<label for="tel type_w">work</label> 
<input type="radio" id="tel_type_o” name=" tel type 
Value="other" /> 
«label for="tel type_o">other</label> 
</fieldset> 


<fieldset> 
<legend>Fax Number:</ legend><br/> 
<input type="text" name="Tax_number" size="30" maxlength="25" 
<input type="radio" id="fax type h" name="fax_ type" 
Value="home" checked /> 
<label for="Tax_type_h">home</label> 
<input type="radio" id="fax type _w" name="fax_ type" 
value="work" /> 
<label for="fax type w">work</label> 
<input type="radio" id="fax_type_o" name="fax_type" 
Value="other" 了 > 
«label for="fax_type_o">other</label> 
</fFfieldset> 


<fieldset> 
<legend>Email Address:</legend><br/> 
<input type="email" name="email" size="3¢@" maxlength="150" /> 
<input type="radio" id="email type_h" name="email type" 
Value="home" checked /> 
<label for="email type h">home</label> 
<input type="radio" id="email_ type _w" name="email_type" 
value="work" /> 
<label for="email type w">work</label> 


i> 


{> 


102: <input type="radio" id="email_type_o" name="email_type" 


103: value= other /> 

104: <label for="email type o">other</label> 
1@5: </fieldset> 

106: 

107: <p><label fTor="note">Personal Note:</label><br/> 
108: <textarea id="note" name="note" cols="35" 

109: rows="3"></textarea></p> 

119: END_OF_TEXT; 

111: if ($ GET} { 

112: $display_block .= "<input type=\"hidden\" name=\"master_id\" 
113: value=\"".$ GET['master_id']."\">"; 

114: } 

113: $display block .= <<<END OF TEXT 

116: <button type="submit" name="submit" 

117: Value="send">Add Entry</button> 

118: </form> 


119: END OF TEXT; 
120: } else if ($_POST) { 


121: /ftime to add to tables, so check for required fields 

122: if ({($_POST['f_name'] == ""} || (S_POST['1 name'] == "")) && 

123: (!isset($ POST['master id']})) { 

124; header( "Location: addentry.php"); 

125: exit; 

126: } 

127: 

128: ¿į iconnect to database 

129: doDB{); 

130: /fcreate clean versions of input strings 

131: $safe f name = mysqli real escape string($mysqli, 

132: $ POST['f name'])}; 

133: $safe 1 name = mysqli real escape string($mysql1li, 

134: $ POST[ Ll name‘ ]}; 

135: $safe address = mysqli real _ escape string(S$mysqli, 

136: $ POST[ 'address']}; 

137: $safe city = mysqli_real_escape_string($mysqli, 

138: $ POST[ 'city‘']); 

139: $safe state = mysqli _real_escape_string($mysqli, 

14@: $ POST[ 'state']); 

141: $safe zipcode = mysqli _ real escape string(Smysqli, 

142: $ POST[ ‘zipcode’ ] ) 

143: $safe tel number = mysqli _real_escape_string{$mysqli, 

144: $ POST[ tel number']); 

145: $safe fax number = mysqli real _ escape string($mysqli, 

146: $ POST[ 'fax_number' ]); 

147: $safe_email = mysgli_real_escape_string($mysqli, 

148: $ POST[ 'email']); 

149: $safe note = mysqli real escape string($mysqgqli, 

158: $ POST[ 'note']}; 

151: 

152: if {!$_POST['master_id']) { 

153: /fadd to master_name table 

154: $add master _ sql = "INSERT INTO master name (date added, 
date_modified, 

155: f name, 1 name) VALUES (now()}, now(}), 

156: '".$safe f name."', '".$safe 1 name."'")"; 


157: $add master res = mysqli query($mysqli, $add master sql) 


158: 
158: 
160: 
161: 
162: 
163: 


164: 
165: 
166: 
167: 
168: 
169: 
17@: 
171: 
1f2: 
173: 
174: 
175; 
176: 
177 
178: 
179: 
180: 
181: 
182: 
183: 
184: 
185: 
186: 
187: 
188: 
189: 
199: 
191: 
192: 
193: 
194: 
195: 
196: 
197: 
198: 
199; 
200: 
201: 
202: 
203: 
204: 
205.: 
206: 
207: 
208! 
209: 
218: 
211: 
212: 


or die(mysqli error{$mysqgli)); 


/fget master id for use with other tables 
$master id = mysqli_insert_id($mysqli); 


} else { 
$master id = mysqli real escape string (Smysqli, 
$$ POST[ 'master_id']); 
} 
if (($_POST['address']) || (S$_POST['city']) || 
($ POST['state']) || (S$ _POST['zipcode'])) { 
//something relevant, so add to address table 
$add address sql = "INSERT INTO address {master id, 


date added, date modified, address, city, state, 
zipcode, type) VALUES 


aaa $master id."", now{), now(), 
.$safe_ address. me "a Ssafe_city."', 
.$safe state. , '".$safe_zipcode."' , 


.$ POST([‘add_type']."°')"; 
ama a es = mysqli query(Smysqli, $add address sql) 
or die(mysqli_error($Smysqli})}; 


} 


if ($ POST['tel number']} 1 
//something relevant, so add to telephone table 


$add tel sql = "INSERT INTO telephone (master_id, date_added, 
date modified, tel number, type) VALUES 
('".$master_id."", now{), now(), 
".$safe tel number."', '".$ POST['tel type']."')"5 


$add tel res = mysqli_query{$mysqli, $add tel sql) 
or die(mysqli error (Smysqli)}; 
} 


if ($_POST['fax_number']) { 
/ ¿something relevant, so add to fax table 


$add fax_sql = "INSERT INTO fax (master_id, date_added, 
date modified, fax number, type) VALUES 
('".Smaster_id."', now{), now(), °'".$safe_fax_number."', 


'".$ POST[ TaX type"j].7')"; 
$add fax res = mysqli_query{$mysqli, $add fax sq1) 
or die(mysqli error(S$mysqli}}); 
I 
if ($_POST[‘email']) { 
//something relevant, so add to email table 


$add email sql = "INSERT INTO email (master_id, date added, 
date modified, email, type) VALUES 


('".Smaster_id."", now{), now({), '".$safe_email."', 
'" $ POST['email type']."')"; 

$add email res = mysqli_ query ($mysqli, add email sql) 
or die(mysqli_error(Smysqli}}; 


} 


if ($_POST['note']} { 
/fsomething relevant, so add to notes table 
$add_notes_sql = "UPDATE personal_notes set note = 
'" $sate note."', date modified = now{) 


213: WHERE master_id = '".$master_id."'"; 


214: } 

eia mysqli close($mysqli)}; 

216: $display block = "<p>Your entry has been added. Would you 
217: like to <a href=\"addentry.php\">add another</a>?</p>"; 
ele. + 

ald: T> 

220: <!DOCTYPE html> 

221: <head> 


222: <title>Add an Entry</title> 
223: </head> 

224: <body> 

225; <h1>Add an Entry</h1> 

226: <?php echo $display block; ?> 
227: </body> 

228: </html> 


AY DAE — ARI OR HF H. A radd info 链 接 来 测试 这 个 修改 后 
的 脚本 。 应 该 会 看 到 如 图 20-7 所 示 的 一 个 页 面 。 


; © Add an Entry x \ 


pa 


| € C | O http://ocalhost/20/addentry?.php?master_id=3 
































Add an Entry 


Adding mformation for John Smith: 


Street Address: 


—City/State/Zip: 





























© home © work © other 


| 局 home E) work © other 

















图 20-7 添加 记录 


在 提 区 了 这 个 表单 后 ， 我 们 可 以 回 到 选择 过 程 并 且 便 看 记录 来 验证 
我 们 已 经 做 出 的 改变 。 


20.8 “4 


在 这 个 动手 实践 的 一 草 中 ， 我 们 应 用 基本 的 PHP 和 MySQL 知 识 来 创 
建 了 一 个 个 人 地 址 矫 。 我 们 学 习 了 如 何 创 建 数 据 库 表 以 及 进行 记录 湛 
加 、 删 除 和 简单 浏览 的 脚本 。 我 们 还 学 习 了 添加 附加 到 一 条 主 条 目的 多 
条 记录 的 过 程 。 


20.9 Q&A 


Q: WRR EE TE HOLE HAS DUA FB, BIA PA 
HAE HRR MS, TE AL? 


A: 不 同 的 表 用 于 地 址 、 电 话 、 传 真 、Email 和 个 人 信息 ， 因 为 一 个 
人 可 能 有 多 条 包含 这 些 信息 的 记录 。 对 于 一 个 人 的 生日 来 说 ， 一 个 人 只 
能 有 一 条 这 样 的 信息 ， 所 以 一 个 天 系数 据 库 可 能 有 些 牛 思 杀 鸡 ， 因 为 每 
个 存在 的 用 户 只 有 一 条 记录 。 因 此 ， 要 添加 一 个 人 的 生日 ， 我 们 应 该 为 
master_name 表 添加 一 个 字段 。 在 为 该 表 添 加 其 他 信息 的 情况 下 ， 首 先 
要 看 一 看 一 个 人 是 只 有 该 信息 的 一 个 实例 《例如 生日 ) ， 还 是 有 多 个 实 
例 〈 例 如 Email 地 址 ) 。 如 条 是 后 者 ， 创 建 一 个 像 address、telephone、 
fax、email 或 personal_notes 这 样 的 表 ， 使 用 master_id 作 为 外 键 。 


20.10 KRAJ 


这 个 实践 练习 设计 用 来 帮助 你 预测 可 能 的 问题 、 复 习 已 经 学 过 的 知 
识 ， 并 且 开 始 把 知识 用 于 实践 。 


1. SNe AT eM PIR, EDIT Oe 
局 变量 ? 


2. 对 于 master_name 表 中 的 每 一 条 单个 的 记录 ， 在 address、email、 
telephone 和 fax 表 中 可 以 有 多 少 条 记录 ? 


3. 通过 哪个 数据 库 字 段 将 额外 的 记录 附加 到 主 记录 


解答 


1 H 局 Ù EA 
e $ G 
S N = 
IN ÍK H AR E 
RS ASA 9 可 以 
任意 
AA AR 9 
ÀJ Y 
这 职 是 


3 
. m 
a 
ster_id ZE 
a 


EA wel 


1. BE H ASF HANNE, MEE eS ae AUER So 
ZN AS BIS AL BERR 


2. (8 Haddentry.php hl Æ W 5 — “SA ASR Tl Ba E P eo eas J 
级 联络 信息 。 图 20-8 展 示 当 二 级 联络 信息 洪 加 给 一 条 记录 之 后 ， 它 是 如 
H ERI o 


| EE 
My Records 


€ C | © http:;//localhost/20/selentry2.php LS 


Showing Record for John Smith 


Addresses: 


e 546 Apple Lane Sometown CA 95128 (home) 
® 999 Blnebird Lane San Jose CA 95128 (work) 


Telephone: 

+ 408-555-1212 (work) 
Fax: 

e 555-555-1313 (work) 
Email: 


* john/@smith com (home) 
* john/@johnsjob.com (work) 


add into ... select another 


120-8 ” 表 中 市 有 多 个 条 目的 一 个 单条 记录 


第 21 瘟 ”创建 一 个 简单 的 讨论 论坛 


在 本 章 中 ， 你 将 学 习 


。 如 何 为 一 个 简单 的 讨论 论坛 创建 表格 。 
。 如 何 为 一 个 简单 的 讨论 论坛 创建 输入 表单 。 
。 如 何 显 示 一 个 简单 的 讨论 论坛 。 


在 本 章 中 ， 我 们 将 要 学 习 一 个 简单 的 讨论 论坛 背后 的 设计 过 程 。 这 
包括 开发 数据 库 表 、 用 户 输入 表单 以 及 显示 结果 。 当 我 们 像 这 样 分 解 讨 
论 论坛 的 时 候 ， 这 样 的 一 个 任务 看 上 去 好 像 很 简单 ， 并 且 实际 上 它 确实 
简单 。 最 终 目 标 是 理解 开发 一 个 像 讨 论 论坛 这 样 的 东西 所 需 的 概念 和 关 
系 ， 而 不 是 要 创建 世界 上 功能 最 完善 的 系统 ， 实 际 上 ， 你 将 看 到 它 并 不 
是 功能 非常 完善 ， 但 它 真 的 是 有 关系 的 。 


21.1 设计 数据 库 表 


考虑 一 个 论坛 的 基本 组 成 部 分 : 主题 和 帖子 。 如 条 论坛 鸭 创立 者 使 
用 正 硝 的 话 ， 一 个 论坛 应 该 包括 几 个 主题 ， 并 且 每 个 主题 应 该 有 用 户 所 
提交 的 一 个 或 多 个 帖子 。 了 解 了 这 一 点 ， 我 们 束 应 该 量 识 到 ， 帖 子 是 通 
过 一 个 键 字 段 联 系 到 主题 的 。 这 个 键 构 成 了 两 个 表 之 间 的 天 系 。 


竹 展 一 下 主题 本 身 的 需求 。 我 们 肯定 需要 一 个 字段 来 保存 标题 ， 并 
且 随 后 可 能 需要 字段 来 你 存 创建 时 间 和 创建 这 个 主题 的 用 户 的 吴 份 。 奖 
MA, A PEERK: 我 们 需要 存储 帖子 的 文本 、 创 建 的 时 间 以 
及 创建 者 。 了 最 重要 的 征 ， 我 们 需要 将 帖子 绑 定 到 主题 的 键 。 


下 面 的 两 个 表 创 建 语句 创建 了 两 个 表 ， 这 两 个 表 叫 做 forum_topics 


Allforum_posts 


CREATE TABLE forum topics ( 
topic_id INT NOT NULL PRIMARY KEY AUTO INCREMENT, 
topic title VARCHAR (158), 
topic_create_time DATETIME, 
topic owner VARCHAR (150) 


7 
CREATE TABLE forum posts ( 
post id INT NOT NULL PRIMARY KEY AUTO INCREMENT, 
topic_id INT NOT NULL, 
post_text TEXT, 
post_create time DATETIME, 
post _owner VARCHAR {150} 





在 这 个 简单 的 论坛 示例 中 ， 我 们 将 通过 用 户 的 Email 地 址 来 识别 用 户 ， 而 不 需要 任何 其 他 
类 型 的 登录 标识 和 从。 





现在 ， 我 们 应 该 有 两 个 空 的 未 等 待 输入。 在 下 一 节 中 ， 我 们 将 创建 
得 入 表单 用 来 添加 一 个 主题 和 一 个 帖子 。 


21.2 为 共同 函数 创建 一 个 包含 文件 


前 面 两 草 为 共同 函数 创建 了 一 个 包含 文件 ， 使 得 脚本 更 为 精简 ， 并 
晶 帮 助 官 理 可 能 随 看 时 间 而 变化 的 信息 ， 例 如 数据 库 用 户 名 和 密码 。 在 
本 章 中 也 是 这 样 。 程 序 清 单 21.1 包 含 了 本 草 中 的 脚本 共 宇 的 代码 。 


程序 清单 21.1 包含 文件 中 的 共同 函数 


1: <?php 

2: function doDB() { 

3: global $mysqli; 

4: 

5. / {connect to server and select database; you may need it 
6: $mysqli = mysqli connect("localhost", "joeuser", 

RE “somepass', "testDB"); 

8: 

9: //1f connection fails, stop script execution 

10: if (mysqli connect_errno()) { 

Be printf ("Connect failed: %s\n", mysqli _connect_error({)); 
T2: exit(); 

13: } 

14: } 

15: Vs 


第 2 行 到 第 14 行 建立 了 一 个 数据 库 连 接 函 数 doDB()。 如 果 没 有 成 功 
建立 连接 ， 在 调用 这 个 函数 的 时 候 ， 脚 本 将 会 退出 ， 人 盏 则 ， 它 将 让 
$mysqli 的 值 可 供 脚 本 其 他 部 分 使 用 。 


把 这 个 文件 保存 为 ch21_include.php 并 将 其 放置 到 Web 服 务 嚣 上。 本 
草 中 的 其 他 代码 将 会 在 脚本 的 前 几 行 中 包含 这 个 文件 。 


21.3 创建 输入 表单 和 脚本 


在 我 们 可 以 添加 任何 帖子 之 前 ， 我 们 必须 癌 论 坛 洪 加 一 个 主题 。 同 
时 添加 一 个 主题 及 该 主题 的 这 一 个 帖子 ， 是 论坛 创建 过 程 中 第 见 的 做 
法 。 从 一 个 用 户 的 观点 来 看 ， 添 加 一 个 主题 然后 返回 ， 然 后 选择 该 主题 
并 添加 一 个 回复 ， 这 么 做 并 没有 多 少 意义 。 我 们 布 户 过 程 尺 可 能 地 平 
顺 。 程 序 清 单 21.2 给 出 了 一 个 创建 新 主题 的 表单 ， 其 中 包含 了 为 主题 中 
的 第 一 个 帖子 留 出 的 空间 。 





程序 清单 21.2 ”添加 一 个 主题 的 表单 


<!DOCTYPE html> 

<html> 

<head> 

<title>Add a Topic</title> 

</head> 

<body> 

<hi>Add a Topic</h1> 

<form method="post" action="do_addtopic.php"> 


On oat OMN 一 


10: <p><label for="topic_owner">Your Email Address:</label><br/> 
11: <input type="email" id="topic_owner" name="topic_owner" size="4Q" 
ta: maxlength="150" required="required" /></p> 


14: <p><label for="topic_title">Topic Title:</label><br/> 

15: <input type="text" id="topic_title" name="topic_title" size="40" 
16: maxlength="150" required="required" /></p> 

17: <p><label for= post text >Post Text:</label><br/> 

18: <textarea id="post_text" name= post text rows="8" 

19: cols="40" ></textarea></p> 


21: <button type="submit" name="Submit" value="Submit">Add Topic</button> 
23: </form> 


24: </body> 
25: </html> 


看 上 去 很 简单 ， 我 们 可 以 从 图 21-1 看 到 ， 表 单 中 出 现 了 3 个 字段 ， 


这 和 古 我 们 需要 在 各 个 表 中 完成 的 ， 你 的 脚本 和 数据 库 可 以 填 与 其 他 的 内 
容 。 把 程序 清单 21.2 保 存 为 类 似 addtopic.html 这 样 的 文件 并 将 其 放置 到 
Webik ax CHIR ASR, UMEDA AT UAIT . 


(©) Add a Topic 
= @ | © http://localhost/21/addtopic.htm 


Add a Topic 


Your Emal Address: 
Topic Title: 


Post Text 





图 21-1 创建 主题 的 表单 


要 在 forum_topics 表 中 创建 条 目 ， 可 以 使 用 来 目 输 入 表单 的 变量 


$_POST[topic title] 和 $_POST[topic_owner] 中 的 值 。topic_id 和 
topic_create_time 字 段 分 别 目 动 增 加 或 通过 MySQL 困 数 now0O 添 加 。 


类 似 地 ， 在 forum_posts 表 中 ， 我 们 使 用 来 目 输入 表单 的 


$ POST['post_text']#$_POST [topic_owner] 中 的 值 ， 而 post_id、 
post_create_time 和 topic_id 字 段 将 会 上 自动 增 加 ， 或 家 提供 。 由 于 我 们 需要 
topic_id 字 段 的 一 个 值 ， 从 而 完成 forum_posts 表 中 的 条 目 ， 我 们 知道 ， 
这 个 得 询 必 须 在 网 forum_topics 表 择 入 记录 的 三 询 之 后 执行 。 程 序 清 单 
21.3 创 建 了 一 个 脚本 来 癌 表 添加 这 些 记 录 。 





程序 清单 21.3 ”添加 一 个 主题 的 脚本 


<?php 
include ‘ch21 include.php' ; 
doDB{}; 


j icheck for required fields from the form 

if ((!$ POST[‘topic_owner']) || (!$_POST[ ‘topic title']) || 
(!$_ POST['post_text'])}) { 
header{ "Location: addtopic.html"); 
exit; 


O N AA BR wh — 


DO 


10: } 


12: //create safe values for input into the database 

13: $clean topic owner = mysqli real escape string($mysqli, 
14: $ POST['topic_owner']); 

15: $clean topic title = mysqli real escape string($mysqli, 


$ POST['topic title']); 


: $clean_post_text = mysqli real escape string($mysqli, 


$ POST[ 'post_text']); 


//create and issue the first query 


: $add topic sql = "INSERT INTO forum topics 


(topic title, topic_create time, topic owner) 
VALUES ('".$clean_topic title ."', now(), 
'", $$clean_topic_owner."')"; 


: $add topic res = mysqli query($mysqli, $add topic sql) 


or die(mysqli_error($mysqli)); 


//get the id of the last query 


: $topic id = mysqli_insert_id($mysql1li) ; 


//create and issue the second query 


: $add post sql = "INSERT INTO forum posts 


(topic_id, post_text, post_create time, post_owner) 
VALUES (*";$topic..id:*’; ’".Selean post text<"*. 
now(), '".$clean_topic_owner."')"; 


: $add post res = mysqli query($mysqli, $add post sql) 


or die(mysqli_error($mysqli)); 
{//close connection to MySQL 


: mysqli _close($mysqli) ; 


//create nice message for user 


: $display block = "<p>The <strong>".$ POST[ "topic title"]."</strong> 


topic has been created.</p>"; 


ee 

> <!DOCTYPE html> 

> <html> 

: <head> 

: <title>New Topic Added</title> 
: </head> 

: <body> 

: <hi>New Topic Added</hi> 

: <?php echo $display block; ?> 
: </body> 

: </html> 


第 2 行 到 第 3 行 包含 了 用 户 创 建 的 函数 的 文件 ， 并 调用 了 数据 库 连 接 


函数 。 接 看 ， 第 6 行 到 第 10 行 检查 要 完成 两 个 表 痢 必需 的 3 个 字段 ，topic 
owner, topic title 以 及 帖子 的 一 些 文本 。 如 采 这 些 字 段 中 的 任何 一 个 不 
人 存在， 用 刀 将 被 重 定 同 到 最 初 的 表单 。 第 13 行 到 第 18 行 创建 了 这 些 变量 


内 容 的 数据 库 安 全 版 本 。 


第 21 行 到 第 27 行 创建 并 插入 了 第 一 个 记录 ， 它 同 forum_topics 表 湛 
加 了 主题 。 注 意 ， 第 一 个 字段 傈 持 衬 日 ， 以 便 上 自动 增加 的 全 可 以 由 每 个 
表 定 义 的 系统 来 添加 。MySQL 的 函数 now0 用 来 在 插入 的 时 候 记 录 下 当 
前 时 间 的 时 间 戳 。 记 录 中 的 其 他 字段 使 用 来 和 目 表 单 的 值 填充 。 


第 30 行 展示 了 一 个 方便 的 函数 mysqli_insert_id0 的 使 用 。 这 个 函数 
可 以 从 这 个 脚本 插入 到 数据 库 中 的 最 后 一 条 记录 中 eaen. EI 
例子 中 ，mysgli_insert_id() 从 forum_topics 表 获取 了 id 值 ， 这 将 成 为 
forum_posts 表 中 的 topic_id 字 上 段 的 值 。 


第 33 行 到 第 39 行 创建 并 插入 了 第 二 个 查询 ， 再 次 使 用 了 已 知 信息 以 
及 系统 提供 信息 的 混合 。 第 二 个 查询 把 用 户 帖 子 的 文本 添加 到 了 
forum_posts 表 。 第 44 行 到 第 45 行 为 用 户 创 建 了 一 个 显示 字符 串 ， 并 且 脚 
本 剩 下 的 内 容 负责 完成 浏览 器 所 要 显示 的 HIML。 


把 这 个 程序 清单 保存 为 do_addtopic.php〔( 即 前 面 的 脚本 所 做 的 动作 
的 名 称 ) 并 且 将 其 放置 到 Web 服 务 需 的 文档 根 目 录 下 。 完 成 网 21-1 创 建 
的 表单 ， 然 后 提交 ， 我 们 应 该 看 到 New Topic Added 消 忌 。 图 21-2 和 21-3 
显示 了 事件 顺序 。 


/ ©) Add a Topic > e A ç O 


Ss O Nttp/focalhost/21/addtopichtmt 























Add a Topic 


Your Emal Address: 


How did you get started with technology? 


I'm interested in knowing how people got 
Started in technology -- did you tinker 

with household electronics? Did you lear 
about it in school? Did your parents buy 
YOu a computer and tell you to have at i 


When did you start hacking on things? 





图 21-2 ”添加 一 个 主题 以 及 第 一 条 帖子 


(©) New Topic Added 


= C |© http://localhost/21/do_addtopic.php Ki A S 


New Topic Added 


The How did vou get started with technology? topic has been created. 


图 21-3 一 个 主题 和 第 一 条 帖子 的 成 功 添加 


在 下 一 市 中 ， 我 们 将 会 把 为 外 两 个 神秘 的 部 分 组 合 到 一 起 ， 显示 主 
题 和 帖子 ， 并 且 回 复 一 个 主题 。 


21.4 ”显示 主题 列表 


既然 在 数据 库 里 有 了 一 个 主题 以 及 至 少 一 个 帖子 ， 我 们 可 以 显示 这 
一 信息 并 且 让 和 人们 洪 加 新 的 主题 或 者 回复 已 有 的 帖子 。 在 程序 清单 21.4 
中 ， 我 们 退回 一 步 并 且 创 建 一 个 页 面 列 出 论坛 中 的 所 有 主题 。 这 个 页 面 
显示 了 每 个 主题 的 基本 信息 并 且 给 用 户 迫 供 一 个 洪 加 新 主题 的 链接 ， 前 
文 已 经 创建 了 添加 新 主题 的 表单 和 脚本 。 程 序 清 单 21.4 中 的 代码 代表 了 
论坛 的 一 个 入 口 页 面 。 


尽 官 程序 清单 21.4 看 上 去 有 很 多 代码 ， 它 实际 上 只 涉及 到 我 们 已 经 
过 到 过 的 很 多 小 的 、 简 时 的 概 翁 ， 从 此 2 行 到 第 3 行 中 的 include() 函 数 和 
DU Ui PEE Pe R BUT UG o 


程序 清单 21.4 主题 列表 脚本 
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<?php 
include ‘ch21 include.php' ; 
doDB{); 


/fgather the topics 

$get_topics sql = "SELECT topic_id, topic title, 
DATE FORMAT(topic create time, sb %e %Y at %r'} AS 
fmt topic create time, topic_owner FROM forum topics 
ORDER BY topic create time DESC"; 


: $get_topics res = mysqli _query($mysqli, $get_topics sql) 


or die(mysgli error($mysqli)}); 


if (mysqli num rows({$get topics res) < 1) { 
//there are no topics, so say so 
$display block = "<p><em>No topics exist.</em></p>"; 
} else { 
f//create the display string 
$display_block <<<END OF _ TEXT 
<table> 
<tr> 
<th>TOPICG TITLE</th> 
<th># of POSTS</th> 
</tr> 
END OF TEXT; 


while ($topic info = mysqli fetch array($get_ topics res)} { 
$topic id = $topic _info[ ‘topic id']; 
$topic title = stripslashes{$topic info['topic title']); 
$topic create time = $topic_info['fmt_topic_create time’ ]; 
$topic _owner = stripslashes($topic_info[ ‘topic_owner']); 


//get number of posts 
$get num posts sql = "SELECT COUNT({post_id) AS post count FROM 
forum posts WHERE topic id = '".$topic_id."'"; 


35: $get_ num posts res = mysqli query{$mysqli, $get num posts sql) 


36: or die(mysqli_error($mysql1)}; 
ns A 

38: while ($posts_info = mysqli fetch array{$get num posts res}} { 
39: $num posts = $posts info['post count']; 

40: } 

41: 

42: iiadd to display 

43: $display block .= <<<END OF TEXT 

44: <tr> 

45: <td><a href="sShowtopic.php?topic id=$topic id"> 
46: <strong>$topic_title</strong></a><br/> 

47: Created on $topic create time by $topic owner</td> 
48: <td class="num_posts_col">$num_posts</td> 

49: <jtr> 

5@: END_OF_TEXT; 

51: } 

Hor {ifree results 

ga: mySqli free result($get topics res); 

54: mysqli free result($get num posts res); 

ro Le 

56: {//close connection to MyS@L 

57: mysqli close($mysqli}; 

58% 

59: {//close up the table 

6a: $display_block .= "</table>"; 

61: } 

62: ?> 

63: <!DOCTYPE html> 

64: <html> 

65: <head> 


66: <title>Topics in My Forum</title> 
67: <style type="text/css"> 


68: table { 

69: border: 1px solid black; 
70: border-collapse: collapse; 
ita } 

72: th { 

i border: 1px solid black; 
74:3 padding: 6px; 

gi font-weight: bold; 

76: background: #ccc; 

tr: } 

ZB: ta + 

79: border: 1px solid black; 
80: padding: 6px; 

81: } 

82: .num_posts_col { text-align: center; } 


83: </style> 

84: </head> 

85: <body> 

86: <hi>Topics in My Forum</h1> 

87: <?php echo $display block; ?> 

88: <p>Would you like to <a href="addtopic.html">add a topic</a>?</p> 
89: </body> 

90: </html> 
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照 日 期 的 降序 来 选取 所 有 的 主题 信息 。 换 句 话 说 ， 按 照 这 样 的 方式 收集 
数据 : 最 近 创 建 的 主题 出 现在 列表 的 项 部。 在 这 个 查询 中 ， 注 意 
date_format() 孙 数 的 使 用 ， 它 将 会 创建 一 个 比 存 储 在 数据 库 中 的 原始 值 
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第 13 行 检 查 了 碍 询 所 返回 的 记录 的 存在 性 。 如 果 没 有 返回 记录 ， 
表明 表 中 没有 主题 ， 我 们 将 希望 告诉 用 户 。 第 15 行 创建 了 这 条 消息 。 此 
时 ， 如 果 不 存在 主题 ， 脚 本 将 会 跳出 证 ...else 结 构 并 且 就 此 结束 :下 一 个 
操作 将 会 在 第 63 行 发 生 ， 也 就 是 静态 HIML 的 开始 。 如 果 脚 本 在 这 里 结 
束 ， 第 15 行 中 创建 的 消息 将 会 在 第 87 行 显示 。 


然而 ， 如 条 forum_topics 表 中 有 了 主题 ， 脚 本 将 在 第 16 行 继续 。 在 
第 18 行 ， 一 段 文 本 赋 给 了 $display_block 变 量 ， 其 中 包含 了 一 个 HTML 表 
格 的 开始 。 第 19 行 到 第 23 行 设置 了 一 个 具有 两 列 的 表格 : 一 列表 示 标 
和 村 ， 一 列表 示 帖 子 的 数目 。 在 第 26 行 ， 我 们 开始 裔 历 查 询 的 结 


第 26 行 中 的 while 循环 确保 有 元 系 从 结 琳 集 中 提取 出 来 ， 把 每 一 
行 作 为 一 个 名 为 $topic_info 的 数组 提取 ， 并 有 晶 使 用 字段 名 作为 数组 元 系 
来 同一 个 新 变量 赋值 。 因 此 ， 在 第 27 行 ， 我 们 要 提取 的 第 一 个 元 系 是 
topic_id 字 7 段 。 把 $topic_info ['topic_id'] 的 值 赋 给 $topic_id 变 量 ， 意 味 看 
我 们 从 名 为 $topic_info 的 数组 中 获得 了 $topic_id 的 一 个 局 部 值 ， 访 数组 
包含 一 个 名 为 topic_id 的 字段 。 继 续 在 第 28 行 到 第 30 行 对 $topic_title、 
$topic_create_time. $topic_owner IX (i. stripslashes() K ZOER V 
在 记录 搬入 的 时 候 输入 到 表 中 的 任何 转 义 字符 。 


在 第 33 行 到 第 36 行 ， 在 while 循 环 之 中 ， 我 们 执行 了 另 一 个 得 询 来 
医 取 某 个 特定 主题 的 帖子 总 数 。 在 第 43 行 ， 我 们 继续 创建 $display_block 
字 从 串 ， 使 用 连接 操作 符 (.=) 来 确保 这 个 字符 串 已 经 附加 到 目前 为 止 已 
经 构建 的 显示 字符 串 的 后 面 。 在 第 45 行 到 第 47 行 ， 我 们 创建 了 HTML 表 
格 来 显示 出 到 showtopic.php 文 件 的 链接 ， 这 个 文件 将 显示 出 主题 以 及 主 
题 的 所 有 者 和 创建 时 间 。 


在 第 48 行 ， 第 二 个 HIML 表 格 列 显 示 了 帖子 的 数目 。 在 第 51 行 ， 我 
们 跳出 了 while 循 环 ， 并 且 在 第 60 行 问 $display_block 字 符 串 添加 最 后 一 
部 分 以 结束 表格 。 剩 下 的 代 但 显示 了 页 面 的 HIML， 包 括 $dqisplay_block 
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如 果 我 们 把 这 个 文件 保存 为 topiclist.php 并 将 其 放置 到 Web 服 务 器 文 
档 根 目 录 下 ， 并 有 旦 如 果 在 数据 库 表 中 有 主题 ， 我 们 会 看 到 如 图 21-4 所 示 
的 结 
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Created on Mar 22 2012 at 01 (05:47 PM By jane(@doe.com 


Would you like to add a topic? 


图 21-4 可 用 主题 


21.5 ”显示 一 个 主题 中 的 帖子 


你 可 能 已 经 猜 到了， 任务 列表 中 的 下 一 项 融 是 构建 showtopic.php 文 
件 来 显示 主题 的 帖子 。 程 序 清 单 21.5 正 是 做 这 些 事情 。 在 这 个 程序 清单 
中 ， 第 6 行 到 第 9 行 检 查 了 GET 碍 询 衬 符 串 中 的 topic_id 的 值 的 人 存在 性 。 
由 于 想 要 时 示 一 个 选 定 的 主题 中 的 所 有 的 帖子 ， 我 们 需要 知道 在 得 询 中 
使 用 哪 一 个 主题 ， 并 且 这 融 是 给 予 我 们 信息 的 方式 。 如 朱 $_GET 
[topic_id] 中 的 值 不 存在 ， 用 户 重 定 同 到 主题 列表 页 面 ， 册 次 等 试 。 


程序 清单 21.5 ”显示 主题 帖子 的 脚本 


| <?php 

2 include 'ch21_include.php' ; 

g: doDB{); 

4: 

si {icheck for required info from the query string 
6: if {!isset({$ GET[ topic id'])) { 

F: header{ "Location: topiclist.php" ); 

p: exit; 

© } 

10: 


11: /f/create safe values for use 
12: $safe topic id = mysqli real escape string($mysqli, $ GET['topic id']); 


14: f/f/verify the topic exists 
15: $verify topic sql = "SELECT topic title FROM forum topics 


16: WHERE topic. id = '" :$safe_ topic id:"'"; 
17: $verify topic res = mysqli query(Smysqli, $verify topic sql) 
18: or die(mysqli error($mysqli)); 


20: if (mysqli num rows ($verify topic res) < 1) { 


21: f/fthis topic does not exist 
22: $display block = "<p><em>You have selected an invalid topic.<br/> 
23! Please <a href=\"topiclist.php\"*>try again</a>.</fem></p>"; 
24: } else { 
25: 1 iget the topic title 
26: while ($topic info = mysqli_fetch_array($verify_topic_res}) { 
27: $topic title = stripslashes($topic_info['topic_title']); 
28: } 
29: 
30: / i gather the posts 
31: $get posts sql = "SELECT post id, post_text, 
DATE FORMAT(post_create_time, 
32: '%b Se %Y<br/>%r') AS fmt_post_create time, post_owner 
33: FROM forum posts 
34: WHERE topic_id = ‘",$safe_topic_id."' 
35: ORDER BY post create time ASC"; 
36: $det posts res = mysqli_query($mysqli, $get_posts_sql) 
37: or die{mysgli_error{$mysgli)); 
38: 
39: //ereate the display string 
AQ: $display block = <<<END GF TEXT 
41: <p>Showing posts for the <strong>$topic title</strong> topic:</p> 
42: <table> 
43: <tr> 
44; <th>AUTHOR</th> 
45: <th>POST</th> 
46: </tr> 
47: END OF TEXT; 
46: 
49: while ($posts_info = mysqli fetch array($get_posts_res)} { 
50: $post id = $posts info['post id']; 
5i: $post text = nl2bristripslashes($posts info['post_text'])}); 
52: $post create time = $posts info['fmt post create time']; 
53: $post_owner = stripslashes({$posts_info['post_owner' ]}; 
54: 
55: fjfadd to display 
56: $display block .= <<<END OF_TEXT 
57: <tr> 
58: <td>$post_owner<br/><br/> 
59: created on:<br/>$post_create_ time</td> 
60: <td>$post_text<br/><br/> 
61: <a href="replytopost.php?past id=$post_ id"> 
62: <strong>REPLY TO POST</strong></a></td> 
63: </tr> 
64: END_OF_TEXT; 
65: } 
66: 
67: 1 /Tree results 
68: mysqli free result($get_ posts res); 
69: mysqli _free_result($verify_topic_res}; 
TO: 
Yi: /fclose connection to MySQL 
T2: mysqli close($mysqli) ; 
fa: 
74: ¿iclose up the table 
75: $display block .= "</table>"; 


76: } 


its W> 

78: <!DOCTYPE html> 

79: <html> 

80: <head> 

81: <title>Posts in Topic</title> 
82: <style type="text/css"> 


83: table { 

84: border: 1px solid black; 
852 border-collapse: collapse; 
86: } 

oT: th { 

88: border: 1px solid black; 
89: padding: 6px; 

90: font-weight: bold; 

91: background: #ccc; 

92: } 

93: td { 

94: border: ipx solid black; 
95: padding: 6px; 

96: vertical-align: top; 

= } 

98: .num posts col { text-align: center; } 
99: </style> 

188: </head> 

191: <body> 


192: <h1>Posts in Topic</h1> 

103: <?php echo $display_ block; ?> 
104: </body> 

185: </html> 


BB 1547 B58 1847 Za th SHEA PA, FPF AIR Se A OR 
证 在 查询 字符 串 中 发 送 的 topic_id 实 际 上 是 一 个 有 效 的 条 目 ， 它 通过 为 
所 涉及 的 主题 选择 相关 的 topic_title 来 做 到 这 一 点 。 如 果 第 20 行 中 的 验证 
和 失效， 将 会 在 第 22 行 到 第 23 行 产生 一 条 消息 ， 并 且 脚 本 将 跳出 证 ...else 语 
句 且 通过 显示 HTML 来 结束 。 输 出 如 网 21-5 所 示 。 
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Posts in Topic 


You have selected an invalid tapie. 
Please iry again. 


图 21-5 ”无 效 的 主题 选取 
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使 用 stripslashes0) 来 删除 任何 转 义 字符 。 接 下 来 ， 在 第 31 行 到 第 37 行 执 
行 一 个 得 询 来 按照 时 间 的 升序 收集 和 该 主题 相关 的 所 有 帖子 。 在 这 个 例 
子 中 ， 最 新 的 帖子 位 于 列表 的 后 部 。 在 第 40 行 ， 开 始 了 一 上段 文 本， 其 中 
包含 了 一 个 HTML 表 格 的 开始 。 第 42 行 到 第 46 行 设置 了 一 个 具有 两 列 的 
表格 : 其 中 一 个 用 于 帖子 的 作者 ， 而 男 一 个 用 于 帖子 文本 本 里 。 我 们 暂 
俘 编 写 文 本 块 ， 并 且 在 第 49 行 开始 过 历 最 初 的 查询 的 结果 。 


第 49 行 的 while 循 环 确保 了 ， 虽 然 有 从 结果 集 提取 的 元 素 ， 但 把 每 
一 行 作为 一 个 名 为 $posts_info 的 数组 来 提取 ， 并 且 把 字段 名 作为 数组 元 
素来 回 一 个 新 的 变量 赋值 。 因 此 ， 在 第 50 行 ， 我 们 试图 提取 的 第 一 个 元 
系 是 post_id 字 段 。 我 们 把 $posts_info [post_id] 的 人 赋 给 了 $post_id 变 
量 ， 意 味 着 我 们 从 一 个 名 为 $posts_info 的 数组 获取 了 S$post_id 的 一 个 局 部 


值 ， 这 个 数组 包含 一 个 名 为 post_id 的 字段 。 在 第 51 行 到 第 53 行 继续 对 
$post_text. $post_create_time#i$post_owneriX A (i. stripslashes() PAi ži 
次 用 来 删除 任何 转 义 字符 ， 并 且 在 $posts_info ['post_text"] 的 值 上 使 用 
nl2br0O 本 数 ， 把 所 有 的 换行 符 奉 换 为 XHTML 兼容 的 换行 符 。 


在 第 56 行 ， 我 们 继续 写 入 $display_block 字 符 串 ， 使 用 连接 操作 符 
(.=) 来 确保 这 个 字符 串 添 加 到 目前 为 止 已 经 创建 的 字符 串 的 末尾 。 在 第 
58 行 到 第 59 行 ， 我 们 创建 了 HTML 表 格 的 一 列 来 显示 帖子 的 作者 和 创建 
时 间 。 在 第 60 行 到 第 63 行 ， 第 二 个 HTML 表 行 显示 了 帖子 的 文本 以 及 一 
个 用 来 回复 帖子 的 链接 。 在 第 65 行 ， 我 们 跳出 了 while 循 环 ， 并 且 在 第 
75 行 回 $display_block 字 符 串 添加 了 最 后 一 部 分 并 且 结 束 了 这 个 表 。 其 有余 
的 代码 显示 了 这 个 页 面 的 HTIML， 包 括 $display_block 字 符 串 的 值 。 


如 果 我 们 把 这 个 文件 保存 为 showtopic.php 并 且 将 其 放置 到 Web 服 务 
名 文档 根 目录 下 ， 如 末 已 经 在 数据 库 中 有 了 帖子 ， 将 会 看 到 如 图 21-6 所 
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Posts in Topic 


Showing posts for the How did vou get started with technology? topic: 


jane(@doe.com | I'm interested in knowmg how people got started in technology -- did you tinker with 


household electronics? Did you learn about it m school? Did your parents buy you a 
created on: computer and tell you to have at it? 
Mar 22 2012 
01:05:47 PM | When did you start hacking on things? 


REPLY TO POST 








图 21-6 一 个 主题 中 的 帖子 
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帖子 的 脚本 来 结束 本 草 。 


21.6 HE eS IA 


在 最 后 的 这 个 步骤 中 ， 我 们 将 创建 replytopost.php 脚 本 ， 其 中 包含 
了 看 上 去 和 用 来 添加 新 主题 的 脚本 关 似 的 代码 。 程 序 清单 21.6 给 出 了 这 
个 一 体 化 的 表单 和 脚本 ， 它 痛 先 在 第 2 行 和 第 3 行 开 始 函 数 文件 的 包含 和 
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还 是 已 经 提交 ) 来 执行 不 同 的 任务 ， 这 两 个 条 件 下 都 需要 在 菏 个 时 刻 和 
数据 库 交 互 。 





程序 清单 21.6” 回 主题 添加 回复 的 脚本 


1: <?php 

2: include ‘ch21_include.php' ; 

3: doDB{); 

4: 

5: jicheck to see if we're showing the form or adding the post 

Bs af #15. POS: { 

T: // showing the form; check for required item in query string 

B: if (!isset($ GET['post id'])) { 

9 : header("Location: topiclist.php"); 

10: exit; 

i des oF 

12: 

13: //create safe values for use 

iå: $safe post_id = mysqli real escape string($mysqli, $ GET['post_id']); 
15: 

16: //still have to verify topic and post 

173 $verify sql = "SELECT ft.topic_id, ft.topic title FROM forum posts 
18: AS fp LEFT JOIN forum topics AS ft ON fp.topic id = 
19: ft.topic id WHERE fp.post id = '".$safe post _id."'"; 


$verify res = mysqli query({$mysqli, Sverify sql) 
or die(mysqli _error($mysqli)); 
if ({mysqli_num_rows($verify_res}) < 1) 4 
f//this post or topic does not exist 
header{"Location: topiclist.php"); 
exit; 
} else { 
/ iget the topic id and title 
while($topic info = mysqli fetch array($verify res}) { 
$topic id = $topic info['topic id']; 
$topic title = stripslashes($topic info['topic title']); 
} 
?> 
<1DOCTYPE html> 
<html> 
<head> 
<title>Post Your Reply in <?php echo $topic title; ?></title> 


</head> 

<body> 

<hi>Post Your Reply in <?php echo $topic title; ?></hi> 

<form method="post" action="<?php echo $_SERVER['PHP_SELF']; ?>"> 
<p><lLabel for="post owner’>Your Email Address:</label><br/> 

<input type="email" id="post_owner" name="post_owner'" size="4@" 
maxlength="15@" required="required'></p> 

<p><label for="post_text">Post Text: </label><br/> 

<textarea id="post text" name="post text’ rows="8" CDOLS= 40 
required="required"></textarea></p> 

<input type="hidden" name="topic id" value="<?php echo $topic id; ?>"> 
<button type="submit" name="submit" value="submit">Add Post</button> 
</form> 


</body> 
</html> 
<?php 
} 
¿ifree result 
mysqli free result($verify_res); 
¿iclose connection to MySQL 
mysqli _close($mysqli); 
} else if ($_POST) { 
/fcheck for required items from form 
1f {{! POST[ topic 1id 1]} || (1$_POST[ post text ]} || 
(!$ POST['post_owner']}} { 
header( "Location: topiclist.php"); 
exit; 
} 
/ioreate safe values for use 
$safe topic id = mysqli_real_escape_string(Smysqli, $_POST[ 'topic_id']); 
$safe post text = mysqli real _escape_string($mysqli, $ POST['post_text']); 
中 Safe post_owner = mysqli _ real _escape_string(Smysqli, $ POST[ post owner ]}; 
/ iadd the post 
$add post sql = “INSERT INTO forum posts (topic _id,post text, 
post_create_time,post_owner} VALUES 
('".$safe topic id."', '".$safe post text."', 


79: now(),'".$safe post _owner."')"; 


809: $add post res = mysgli query($mysqli, $add post sql) 
815 or die(mysqli_ error($mysqli) ); 

82: 

83: //close connection to MyS@L 

84: mysqli close($mysqli}; 

85; 

86: //redirect user to topic 

or: header{"Location: showtopic.php?topic id=".$ POST['topic id']}; 
88: exit; 

89: } 

909: ?> 


第 6 行 检查 表单 是 否 已 经 提交 。 如 果 $_POST 还 没有 值 ， 表 明 表 单 还 
没有 提交 ， 并 且 必 须 显示 它 。 然 而 ， 在 显示 表单 之 前 ， 必 须 检查 一 个 必 
需 的 项 目 ， 第 8 行 到 第 11 行 检查 GET 查 询 字符 串 中 的 post_ id 是 否 有 一 个 
值 存在 。 如 果 $_GET[post_id] 中 没有 值 存在 ， 用 户 重 定向 到 主题 列表 页 
if. 


Qn RELI F$_GET['post_id'] F MERA S, 21777 BER 22 
行 执行 一 个 看 似 复杂 的 查询 ， 它 根据 我 们 所 知道 的 唯一 的 值 
($_GET[post id] 的 值 ) \forum_topics#< 3k Htopic_id#ltopic_title~: Ex 
的 仁 。 这 个 得 询 既 验证 了 帖子 的 存在 性 ， 也 获取 了 我 们 在 后 面 的 脚本 中 
将 要 用 到 的 信息 。 第 24 行 到 第 27 行 根据 这 一 验证 测试 的 结 末 来 执行 ， 如 
果 测 斌 失败 ， 融 将 用 户 重 定 同 到 topiclist.php 页 面 。 


如 来 $_GET['post_id'] 的 值 表 示 一 个 有 效 的 帖子 ， 我 们 在 第 30 行 到 第 
3347 $e Mtopic_id#ltopic_titleA(H, FEV Astripslashes() A BOK Mt BRE 
何 转 义 字符 。 接 下 来 ， 用 来 添加 一 个 帖子 的 表单 显示 在 整个 屏幕 上 ， 直 
到 点击 提交 控 钮 ， 这 个 脚本 束 古 这 样 。 在 这 个 表 时 中 ， 我 们 看 到 动作 十 
在 第 42 行 的 $_SERVER['PHP_SELF']， 表 示 这 个 脚本 将 再 次 调用 到 动作 


执行 中 。 第 49 行 的 一 个 隐藏 字段 保存 了 需要 传递 到 脚本 的 下 一 次 迁 代 中 


的 信息 。 


转 癌 第 62 行 ， 当 脚本 重新 载 入 并 且 $_POST 包 含 一 个 值 的 时 候 ， 这 
段 代 码 将 会 执行 。 这 上段 代码 检 可 了 来 目 表 单 的 所 有 必需 的 字段 的 存在 
(第 64 行 到 第 68 行 ) ， 然 后 ， 如 果 它 们 都 存在 ， 使 用 第 71 行 到 第 73 行 所 
创建 的 安全 值 ， 提 交 僵 询 来 同 数 据 库 洪 加 帖子 (第 76 行 到 第 81 行 )。 妆 
帖子 洪 加 a 到 数据 库 之 后 ， 苗 定 同 到 showtopic.php 幢 面 〈 第 87 行 到 第 88 
行 )， 使 用 相应 的 查询 字 从 串 来 显示 活动 的 主题 。 


如 果 我 们 把 这 个 文件 保存 为 replytopost.php 并 且 将 其 放置 到 Web 服 务 
古文 档 根 目录 下 ， 目 己 笠 试 后 将 会 看 到 如 疼 21-7 和 图 21-8 所 示 的 结 


. 吕 | E 
© Fost Your Reply in How did 
全 CG | © http://localhost/21/replytopost.php?post id=2 Ki KE x 


Post Your Reply in How did you get started 
with technology? 


Your Email Address: 


Post Text: 


图 21-7 ”准备 添加 一 个 帖子 


© Posts in Topic 


T © | © http;/localhost/21/showtopic.php?topic_id=2 Kl 所 S 


Posts in Topic 


Showing posts for the How did vou get started with technology? topic: 


jane(@doe.com 


created om 
Mar 22 2012 
01:05:47 PM 


anna(@able.com 


created om 
Mar 22 2012 
10:50:49 PM 


I'm interested in knowing how people got started in technology -- did you tinker 
with household electronics? Did you learn about it m school? Did your parents buy 
you a computer and tell you to have at it? 


When did you start hacking on things’? 


REPLY TO POST 


My parents gave me a TRS-80 when I was 11, and I played all sorts of text 
adventures and learned a little BASIC. I always found computers fascmating, and 
used them whenever I could -- even when my school didn't! 


REPLY TO POST 





图 21-8 添加 到 列表 的 一 个 帖子 


Di Ne 


要 采取 从 初学 到 熟练 的 思想 ， 我 们 十 要 巡 循 一 个 设计 过 程 。 这 个 设 
计 过 程 基本 上 可 以 概括 为 “ 先 想 后 做 ”*。 讨 论 规则 、 和 需求 和 目标 ， 然 后 创 
建 规范 化 的 表 的 最 终 版 本 。 


在 本 革 ， 我 们 看 到 了 论坛 为 什么 本 质 上 是 层级 的 : 论坛 包括 主题 ， 
主题 包括 帖子 。 我 们 不 能 拥有 一 个 没有 帖子 的 主题 ， 并 且 帖 子 也 不 会 存 
在 于 不 属于 任何 主题 的 论坛 中 。 我 们 把 这 一 知识 应 用 到 用 来 存储 论坛 主 
题 和 帖子 的 表 的 创建 中 ， 并 且 使 用 PHP 脚 本 来 创建 这 些 项 的 输入 和 输出 
页 和 面 。 


21.8 Q&A 


Q: 如 果 我 们 想 要 有 多 个 论坛 ， 该 怎么 办 ? 假设 只 有 一 个 论 
坛 系统 可 用 。 


A: 如 果 我 们 想 在 讨论 板块 上 拥有 多 个 论坛 ， 创 建 一 个 名 为 forums 
或 者 共有 类似 效果 的 名 字 的 表 ， 它 包含 一 个 ID 字段 、name 字 段 并 且 可 
能 还 有 一 个 论坛 说 明 字 段 。 然 后 ， 在 forum_topics 和 forum_posts 表 中 ， 
次 加 一 个 名 为 forum_id 的 字段 ， 以 便 这 些 在 层级 中 较 低 的 元 系 可 以 绑 定 
到 主 论坛 。 确 保修 改 用 来 插入 记录 的 SQL 语 铅 ， 考 虑 到 forum_id 的 什 。 


接 下 来 ， 从 论坛 层级 开始 显示 ， 而 不 是 从 主题 层级 开始 。 就 像 我 们 
不 能 先 创 建 一 个 显示 主题 的 脚本 ， 然 后 再 创建 一 个 显示 论坛 的 脚本 。 到 
论坛 显示 的 链接 应 该 包含 forum_id， 并 且 页 面目 里 应 该 在 该 论坛 中 显示 
所 有 的 主题 。 


21.9 RAS 


这 个 实践 练习 设计 用 来 帮助 你 预测 可 能 的 问题 、 复 习 已 经 学 过 的 知 
识 ， 并 且 开 始 把 知识 用 于 实践 。 


问 容 题 


1. 主题 ID 的 全 是 如 何 传递 到 showtopic.php 脚 本 中 的 ? 


2. 除了 告诉 用 户主 题 极 成功 地 添加 了 ， 在 do_addtopic.php 脚 本 的 末 
尾 我 们 还 能 做 什么 ? 


3. 这 上 段 脚 本 为 何 根据 表单 中 的 值 使 用 mysqli_real_escape_string() 也 | 
数 。 


Fa 
oy 


1. 通过 超 全 局 变量 $_ GET, BV$ GET['topic_ id] 的 值 。 


2， 束 像 replytopostphp 脚 本 一 样 ， 我 们 可 以 删除 消息 显示 而 二 接 把 
用 户 重 定 同 到 她 刚刚 创建 的 主题 ， 受 示 新 主题 及 其 中 的 所 有 帆 子 。 


3. mysqli_real_escape_stringO) 函 数 通 过 准备 “安全 的 ”字符 串 以 将 其 
插入 到 数据 库 表 中 ， 从 而 预防 SQL 注入 陈 攻 击 。 


EA wel 


1. 你 将 注意 到 没有 页 面 会 真正 地 和 任何 类 型 的 导航 绑 定 到 一 起 。 
获取 这 些 基 本 的 框 染 脚 本 并 且 将 一 些 导 般 流 程 用 到 其 中 。 确 保 用 户 总 十 
可 以 洪 加 一 个 主题 或 者 从 任何 给 定 的 页 面 返回 到 主题 列表 。 


2. 如 来 你 感到 意犹未尽 ， 使 用 Q&A 部 分 提供 的 信息 来 把 多 个 论坛 
整合 并 显示 到 紧 旋 而 小 巧 的 讨论 板 其 中 。 当 你 这 么 做 的 时 候 ， 应 用 一 些 
文本 样式 和 颜色 ， 给 原本 简陋 的 示例 深 加 一 些 腕 


第 22 章 ”创建 一 个 在 线 商 店 


© 为 一 个 在 线 商 店 创建 天 系 表 。 
。 创建 用 来 显示 商品 分 类 的 脚本 。 
。 Gl) Sa sme ANTS Fal m HY HAA 


在 这 个 简短 的 动手 实践 诛 程 中 ， 我 们 将 创建 一 个 通用 的 在 线 丙 店 。 
你 将 学 会 创建 相关 数据 库 表 的 方法 ， 以 及 用 来 为 用 户 显 示 信 息 的 脚本 。 
本 和 草 中 使 用 的 这 个 例子 表示 了 完成 这 些 任 务 的 无 数 多 种 可 能 性 方法 中 的 
一 种 ， 并 且 本 例 用 来 为 你 提供 基础 的 知识 ， 而 不 是 完成 这 一 任务 的 确定 
方法 。 


22.1 规划 和 创建 数据 库 表 


在 我 们 为 一 个 在 线 商 店 处 理 创 建 数据 库 的 过 程 之 前 ， 考 虑 一 下 现实 
生活 中 的 购物 过 程 。 当 我 们 走 进 一 家 商店， 商品 都 是 按照 示 种 顺序 排 
列 : 计算 机 便 件 和 轻 儿 的 衣服 是 不 会 放 在 一 起 的 ， 电 帮 和 和 洗衣 粉 个 会 近 
看 控 放 ， 等 等 。 把 这 些 知 识 应 用 到 数据 库 规 施 化 中 ， 你 束 能 明日 害 要 一 
个 表 来 存储 分 类 ， 一 个 表 来 存储 商品 。 在 这 个 简单 的 丙 店 中 ， 所 有 商品 
ANB TATRA 


接 下 来 ， 考 碟 一 下 商品 本 喘 。 根 据 我们 所 拥有 的 商店 的 闫 型 ， 商 品 
可 能 有 也 可 能 没有 颜色 ， 可 能 有 大 小 也 可 能 没有 大 小 。 但 是 ， 所 有 的 丙 
后 部 有 一 个 名 字 、 一 个 说 明 以 及 一 个 价格 。 我 们 再 一 人 次 考虑 规范 化 的 问 
题 ， 你 可 以 看 到 目 己 拥有 一 个 通用 商品 表 以 及 和 通用 丙 品 表 相 关 的 两 个 
其 他 的 表 。 


表 22-1 显 示 了 在 线 商店 的 示例 表 和 字段 名 。 只 需要 一 分 钟 的 时 间 ， 
束 可 以 编写 实际 的 SQL 语 名 ,但 是 ， 痛 完 你 需要 看 看 这 些 信息 并 且 壬 试 
看 看 其 大 系 。 目 己 思 考 一 下 哪个 字段 应 该 是 主键 或 唯一 键 。 


表 22-1 商店 表 和 字段 名 





id, cat_title, cat_desc 


store_items id, cat_id, item_title, item_price, item_desc, item_image 


store item size item_id, item_size 





store item color item_id, item_color 


正如 我 们 将 在 如 下 的 SQL 语句 中 看 到 的 ， 除 了 id 字段 以 外 ， 
store_categories 表 还 拥有 两 个 其 他 的 字段 : cat_titte 和 cat_desc， 用 来 存储 
标题 和 说 明 。id 字 段 是 主键 ， 而 cat_title 是 一 个 唯一 键 ， 因 为 ， 没 有 理由 
拥有 两 个 相同 的 分 类 。 
CREATE TABLE store categories ( 
id INT NOT NULL PRIMARY KEY AUTO_INCREMENT, 


cat title VARCHAR (50) UNIQUE, 
cat desc TEXT 


); 


接 下 来 ， 我 们 要 处 理 store_items 表 ， 除 了 id 字段 ， 它 还 拥有 5 个 字 
段 ， 没 有 一 个 是 唯一 键 。 字 上 段 定 义 中 指定 的 长 度 是 任意 的 ， 你 应 该 使 用 
最 适合 自己 的 大 小 。 


cat_id 字 段 把 商品 和 store_categories 表 中 的 一 个 特定 分 类 关联 起 来 。 
这 个 字段 不 是 唯一 的 ， 因 为 我 们 希望 每 个 分 类 有 多 个 商品 。item _title、 
item_price 和 item_desc 《用 于 说 明 ) 字段 都 是 一 日 了 然 的 。item_image 字 
段 将 保存 一 个 文件 名 ， 在 这 个 例子 中 ， 假 设 文 件 是 服务 颖 的 本 地 文件 ， 
当 需 要 显示 商品 信息 的 时 候 ， 我 们 用 这 个 文件 名 来 构建 一 个 HTML 
<img> 标 记 。 


CREATE TABLE store_items ( 
id INT NOT NULL PRIMARY KEY AUTO INCREMENT, 
cat_id INT NOT NULL, 
item_title VARCHAR (75), 
item price FLOAT (8,2), 
item desc TEXT, 
item_image VARCHAR (56) 


store item sizefllstore item color 表 包含 了 可 选 的 信息 : 如 果 我 们 销 
售 图 书 ， 图 书 是 不 会 有 大 小 和 闫 色 的 ， 但 如 果 销 售 衬 衫 ， 它 们 就 有 大 小 
和 闫 色 了 。 对 于 这 些 表 中 的 每 一 个 ， 痢 不 包含 键 ， 因 为 我 们 可 以 把 任 总 
多 种 颜色 和 大 小 与 一 个 特定 商品 关联 起 来 。 


CREATE TABLE store_item_size ( 

id INT NOT NULL PRIMARY KEY AUTO INCREMENT, 

item_id INT NOT NULL, 

item size VARCHAR (25) 
CREATE TABLE store_item_color { 

id INT NOT NULL PRIMARY KEY AUTO_INCREMENT, 

item_id INT NOT NULL, 

item color VARCHAR (25) 
E 

WA SMETANA Prim ATA, Bew, HRE 

示 归 销售 的 商品 。 第 23 草 把 用 户 体 验 集成 了 进来 。 到 现在 ， 我 们 只 需要 


把 注意 力 集 中 在 两 品目 录 上 。 


在 第 20 章 中 ， 我 们 学 习 了 如 何 使 用 PHP 表 单 和 脚本 来 添加 或 删除 表 
中 的 记录 。 如 果 我 们 把 相同 的 原理 应 用 到 这 一 组 表 上 ， 可 以 很 容易 地 创 
建 一 个 对 商店 的 管理 前 闪 。 我 们 在 本 书 中 不 会 介绍 这 一 过 程 ， 但 你 可 以 
自己 去 尝试 这 么 做 。 在 这 里 ， 我 们 只 是 告诉 你 ， 你 学 习 的 PHP 和 MySQL 
的 知识 已 经 足够 完成 这 个 任务 。 


现在 ， 我 们 可 以 通过 MySQL 监 视 器 或 其 他 的 界面 来 简单 地 执行 


MySQL 俘 询 ， 同 表 中 添加 信息 。 如 下 十 一 些 例 子 ， 如 末 你 想 要 使 用 示 
例 数 据 的 证。 


22.1.1 问 store_categories 表 插入 记 采 


如 下 的 查询 在 store_categories 表 中 创建 了 3 个 分 类 : hats、shirts 和 
books. 


INSERT INTO store_categories VALUES 
('1', ‘Hats', ‘Funky hats in all shapes and sizes!'); 


INSERT INTO store_categories VALUES ( 2 ， ‘Shirts’, ‘From t-shirts to 
sweatshirts to polo shirts and beyond.'); 


INSERT INTO store_categories VALUES ('3', ‘Books’, ‘Paperback, hardback, 


books for school or play.'); 


在 下 一 节 中 ， 我 们 将 向 分 类 中 添加 丙 品 。 
22.1.2 ” 问 store items 表 搬入 记 杂 
如 下 的 碍 询问 每 个 分 类 中 添加 了 3 种 商品 。 可 以 任意 添加 更 多 丙 


器 
HH o 


INSERT INTO store_items VALUES ('1', '1', ‘Baseball Hat', 12.00 ， 
‘Fancy, low-profile baseball hat.', ‘baseballhat.gif'); 


INSERT INTO store _items VALUES (2 ， 1 ， ‘Cowboy Hat', ‘52.00’, 
'10 gallon variety’, ‘cowboyhat.giT'}; 


INSERT INTO store_items VALUES ( 3 ， 1 ， Top Hat', '182.00', 
‘Good for costumes.', ‘tophat.gif'); 


INSERT INTO store_items VALUES ('4', ‘'2', ‘Short-Sleeved T-Shirt’, 
'12.00', '100% cotton, pre-shrunk.', ‘sstshirt.gif'}; 


INSERT INTO store items VALUES ('5', ‘2°, ‘Long-Sleeved T-Shirt’, 
'15.00°, ‘Just like the short-sleeved shirt, with longer sleeves.', 
‘lstshirt.gif'}; 


INSERT INTO store_items VALUES ('6', ‘2', ‘Sweatshirt’, ‘22.00’, 
'Heavy and warm.', ‘sweatshirt.gif'}; 


INSERT INTO store items VALUES ('7', ‘3', ‘Jane\'s Self-Help Book’, 
'12.09', ‘Jane gives advice.', ‘selfhelpbook.gif'); 


INSERT INTO store_items VALUES (‘'8', '3', ‘Generic Academic Book , 
'35.00', ‘Some required reading for school, will put you to sleep.', 
‘horingbook .gif'); 


INSERT INTO store items VALUES ('9', '3', ‘Chicago Manual of Style', 
'9.99', ‘Good for copywriters.', ‘chicagostyle.gif'}; 


提示 : 








上 面 的 查询 引用 了 和 名称 英 似 于 “baseballhat.gif 的 各 种 图 形 ， 而 这 些 图 形 并 没有 包 舍 在 代码 
中 。 你 可 以 目 己 找 一 些 示例 图 像 或 目 己 制作 一 些 占 位 符 图 形 。 


22.1.3 |AJstore item size 表 中 插入 记录 


下 和 面 的 查询 把 大 小 和 shirts 分 类 中 的 3 种 商品 中 的 一 种 天 联 起 来 ， 并 
且 把 一 个 通用 的 “one size fits all” GJ) 的 大 小 和 hats 分 类 中 的 每 一 种 
商品 关联 起 来 〈 假 设 这 些 帽 子 都 是 奇怪 的 帽 于 ) 。 请 你 目 行 对 shirts 分 类 
中 的 其 他 商品 插入 相关 的 同一 组 大 小 。 


INSERT 
INSERT 
INSERT 
INSERT 
INSERT 
INSERT 
INSERT 


INTO 
INTO 
INTO 
INTO 
INTO 
INTO 
INTO 


store item size 
store item size 
store item size 
store item size 
store item size 
store item size 
store item size 


(item_id, 
(item_id, 
(item_id, 
(item_id, 
(item_id, 
(item_id, 
(item_id, 


item_size) 
item_size) 
item_size) 
item_size) 
item_size) 
item_size) 
item_size) 


VALUES 
VALUES 
VALUES 
VALUES 
VALUES 
VALUES 
VALUES 


22.1.4 |Hjstore item color 表 插入 记 孙 


'One Size Fits All'); 
'One Size Fits All'); 
'One Size Fits All'); 
S'); 
M’); 
L’); 
'XL'); 


如 下 的 三 询 把 两 色 和 shirts 分 类 中 的 3 种 商品 中 的 一 种 关联 起 来 。 请 
目 行 为 其 他 的 Shirts 和 hats 插 入 凑 色 记录 。 


INSERT INTO store item color (item_id, item_color) VALUES (1,'red'); 


INSERT INTO store item color (item_id, item_color) VALUES (1, ‘black'); 
INSERT INTO store item color (item_id, item_color) VALUES (1, '‘blue'); 


IE Re BA as KS A EES UE A ET MER 
分 类 及 商品 相 比 ， 创 建 用 来 显示 信息 的 脚本 更 加 容易 。 我 们 将 要 创建 的 
第 一 个 脚本 用 来 列 出 分 关 和 商品 。 蛙 然 ， 我 们 不 希望 在 用 户 刚 一 进门 的 
时 候 就 列 出 所 有 的 分 类 和 商品 ， 但 是 确实 希望 给 用 户 一 个 选择 能 够 并 即 
选取 分 类 、 俘 看 商品 ， 并 且 随 后 选择 其 他 分 类 。 换 句 话 说 ， 这 个 脚本 有 
两 个 目的 : 它 显 示 分 类 ， 然 后 如 下 用 尸 后 击 一 个 分 类 连接 ， 它 显示 坟 分 
关中 的 商品 。 


程序 清单 22.1 给 出 了 seestore.php 的 代码 。 如 来 你 按照 顺序 阅读 本 
书 ， 将 会 注意 到 ， 在 前 面 各 草 中 有 很 多 同样 的 基本 结构 ;正如 我 在 本 书 
前 面 提 到 的 ， 这 些 项 目 是 基本 的 CRUD (创建 、 读 取 、 更 新 、 删 除 〉 应 
用 程序 的 所 有 示例 。 即 便 如 此 ， 在 程序 清单 之 后 ， 我 们 还 是 详细 解释 了 
人 





程序 清单 22.1 浏览 分 类 的 脚本 
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<?php 
//cannect to database 


} else { 
while ($cats = mysqli fetch array($get cats res)) { 
$cat_id = $cats['id']; 


Smysqli = mysqli connect( "localhost", "joeuser", "somepass", "testDB"}; 
$display block = "<hi>My Categories</h1> 
<p>Select a category to see its items.</p>'; 
//show categories first 
$get_cats_sql = "SELECT id, cat_title, cat_desc FROM 
store categories ORDER BY cat title"; 
: $get_cats_res = mysqli _query($mysqli, $get_cats sql) 
or die({mysqli error($mysqli)); 
if (mysqli num rows($get cats res) < 1) { 
$display block = "<p><em>Sorry, no categories to browse.</em></p>"; 


$cat title = strtoupper(stripslashes({$cats[ ‘cat title']}); 


$cat desc = stripslashes($cats['cat_desc']}; 


$display block .= "<p><strong><a href=\"".$ SERVER[ 'PHP SELF']. 
"?cat_id=".$cat_id.*"\">".$cat_title."</a></strong><br/>" 


.$cat desc."</p>"; 


if (isset($ GET[ cat 1d ]) && ($ GET[ cat 1d ] == $cat_id}) { 


rrcreate safe value for use 
$safe cat id = mysqli _real_escape_string{$mysqli, 
$ GET[ cat id ]); 


//get items 
$get_items sql = "SELECT id, item_title, item_price 
FROM store items WHERE 
cat_id = ‘'".$cat_id."’ ORDER BY item title"; 
$get items res = mysqli query({$mysqli, $get items sql} 
or die(mysqli_error($mysqli)); 


if (mysqli_num_rows($get_items_res) < 1) { 


$display block = "<p><em>Sorry, no items in this 
category.</em></p>"; 

} else { 
$display_ block .= "<ul>"; 


while ($items = mysqli fetch array({$get_ items res)) { 
$item id = $items['id']; 
$item title = stripslashes($items[ ‘item title’ ]); 
$item price = $items['item price']; 


$display block .= "<li><a 
href=\"show1ltem.php?item id=". 

$item id."\">".$item title. "</a> 

(\$".$1item_price.")</li>"; 


} 


$display_ block .= "</ul>"; 
} 
//free results 
mysqli free result($get items res); 


/ {free results 

: mysqli free result($get cats res); 
/ {close connection to MYSQL 

: mysqli close($mysqli); 


: <!DOCTYPE html> 


69: <title>My Categories</title> 
70: </head> 
71: <body> 
72: <?php echo $display block; ?> 
73: </body> 
74: </html> 


既然 我 们 在 第 20 草 看 到 过 更 长 的 代码 ， 这 个 74 行 的 功能 完整 的 代码 
束 比 较 容 易 接 党 一 些 。 在 第 3 行 ， 先 打开 了 数据 库 连 接 ， 因 为 不 官 脚本 
要 采取 什么 动作 《显示 分 类 或 者 显示 分 类 中 的 商品 ) 都 需要 数据 库 。 


在 第 5 行 ， 开 始 了 $display_block 字 人 符 串 ， 将 一 些 基 本 的 页 面 标题 信 
轧 添 加 到 其 中 。 第 9 行 到 第 12 行 创建 并 执行 了 三 询 来 获取 分 类 信息 。 人 第 
14 行 检查 分 类 ， 如 果 表 中 没有 分 类 ， 在 $display 块 变量 中 存储 一 条 消 
思 ， 以 便 同 用 户 显 示 ， 这 焉 是 这 个 脚本 所 做 的 所 有 的 事情 《〈 它 跳 转 到 第 
66 行 的 HIML， 并 且 在 衬 出 一 些 数据 库 结 琳 之 后 打印 到 屏幕 上 ) 。 然 
而 ， 如 末 找 到 分 类， 脚本 移动 到 第 17 行 ， 开 始 一 个 while 循 环 来 提取 分 


ARAB E o 


在 这 个 while 循 环 中 ， 第 18 行 到 第 20 行 获取 ID、 标 题 和 分 类 的 说 
明 。 执 行 了 字符 串 操 作 以 硝 保 文 本 中 没有 笠 杠 并 且 分 类 标题 是 大 与 格 
式 ， 以 便 显 示 。 第 22 行 到 第 24 行 放置 分 类 信息 ， 包 括 一 个 目 引 用 的 页 面 
链接 ， 位 于 $display_block 字 符 串 中 。 如 果 用 户 点 击 了 这 个 链接 ， 她 将 返 
回 这 个 脚本 ， 只 不 过 将 一 个 分 类 ID 在 查询 字符 串 中 传递 。 这 个 脚本 在 第 
26 行 检查 这 个 值 。 


如 采用 户 和 希望 簿 看 商品 而 点 击 了 一 个 分 类 连接 ， 从 而 使 
$_GET[cat_id] 的 值 传递 给 了 脚本 ， 这 个 脚本 将 会 构建 并 执行 另 一 个 得 
询 ( 第 32 行 到 第 36 行 ) 来 获取 此 分 类 中 的 商品 。 第 38 行 到 第 51 行 检 僵 了 
了 商品， 然后 构建 了 一 个 商品 字符 串 作 为 $display_block 的 一 部 分 。 了 字符 串 


中 的 部 分 信息 是 到 一 个 名 为 showitem.php 的 脚本 的 链接 ， 这 个 脚本 将 在 
下 一 市 创建 。 


到 这 里 之 后 ， 脚 本 所 要 做 的 事情 束 不 多 了 ， 它 显示 了 HTML 以 及 
$display_block 的 值 。 图 22-1 显 示 直 接 访 问 脚 本 时 它 的 输出 ， 只 有 分 类 信 
FASE AN LOR 
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© My Categories 


= @ | © http://localhost/22/seestore.ohp Kl E x 


My Categories 


Select a category to see its tems. 


BOOKS 
Paperback, hardback, books for school or play. 


HATS 
Funky hats in all shapes and sizes! 


SHIRTS 
From t-shirts to sweatshirts to polo shirts and beyond. 








图 22-1 ”商店 中 的 分 类 


在 图 22-2 中 ， 我 们 看 到 了 当 用 户 点 击 HATS 链 接 的 时 候 会 发 生 什 
A: 脚本 收集 了 和 访 分 类 相关 的 所 有 商品 并 且 将 它们 显示 在 屏幕 上 。 用 
户 仍然 可 以 跳 转 到 同一 个 页 面 上 的 另 一 个 分 类 ， 并 且 它 将 收集 该 分 类 的 
所 有 商品 。 


My Categories 


ée C | © htto;//localhost/22/seestore.php?cat id=1 


My Categories 
Select a category to see its tems. 


BOOKS 
Paperback, hardback, books for school or play. 


HATS 
Funky hats in all shapes and sizes! 


* Baseball Hat ($12.00) 
è Cowboy Hat ($52.00) 
e Top Hat ($102.00) 


SHIRTS 
From t-shirts to sweatshirts to polo shirts and beyond. 


图 22-2 商店 的 一 个 分 类 中 的 商品 


本 章 的 难题 的 最 后 一 部 分 是 商品 坚 示 页 面 的 创建 。 


22.3 SANTA on 


本 章 中 的 商品 显示 页 面 只 是 显示 了 所 有 商品 信息 。 在 第 23 草 中 ， 我 
们 将 向 其 中 添加 一 些 代码 ， 使 其 带 有 一 个 cadd to cart" 按 钮 的 功能 。 程 序 
清单 22.2 显 示 了 showitem.php 的 代码 。 


程序 清单 22.2 ”浏览 商品 信息 的 脚本 


COND Oh WN — 


© 


18: 
19: 
20: 
PA E 
22: 
fo: 
24: 
Eg: 
eu 
ef: 
28: 
29: 
3@: 
si: 
32: 
333 
34: 
Sas 


36: 


37: 
38: 
39: 
40: 
41: 
42: 
43: 
44: 


<?php 

/{/connect to database 

$mysqli = mysqli connect("“localhost", "joeuser", "Somepass", "testDB"); 
$display block = "<hi>My Store - Item Detail</h1>'; 


/{create safe values for use 
$safe item id = mysqli real escape string{$mysqli, $ GET['item id']}); 


//validate item 


: $get_item_sql = "SELECT c.id as cat_id, c.cat_title, si.item_title, 


Si.item price, si.item desc, si.item image FROM store items 
AS si LEFT JOIN store categories AS c on c.id = si.cat id 
WHERE si.id = '".$safe_item_id."''"; 


: $get_item_res = mysqli _query($mysqli, $get_item_sql) 


or die(mysqli error{($mysqli)); 


if (mysqli num rows($get item res) < 1) { 
//invalid item 
$display block .= "<p><em>Invalid item selection.</em></p>"; 
} else { 
//valid item, get info 
while ($item info = mysqli fetch array({$get_ item res)) { 
$cat id = $item_info['cat_id']; 
$cat title = strtoupper(stripslashes($item info['cat title'])); 
$item title = stripslashes($item_info['item_title']); 
$item price = $item info['item price']; 
$item_desc = stripslashes($item_info['item_desc']); 
$item image = $item info['item image']; 


} 


//make breadcrumb trail & display of item 

$display_block .= <<<END OF_TEXT 

<p><em>You are viewing: </em><br/> 

<strong><a href="seestore.php?cat_id=$cat_id">$cat_title</a> &gt; 
$item title</strong></p> 

<div style="float: left;"><img src="$item_image" alt="$item_ title" 
{/></div> 

<div style="float: left; padding-left: 12px"> 

<p><strong>Description: </strong><br/>$item desc</p> 

<p><strong>Price:</strong> \$$item_price</p> 

END OF TEXT; 


//free result 
mysqli_free_result($get_item_res) ; 


45: ffget colors 


46: $get_colors sql = "SELECT item color FROM store item color WHERE 
47: item_id = '".$safe_item_id."' ORDER BY item color"; 
48: $get_colors res = mysqli _query($mysqli, $get_ colors sql} 

49: or die(mysqli error($mysqli)); 

5@: 

zi ee if (mysqli_num_rows($get_colors res) > 0) { 

52: $display block .= "<p><strong>Available Colors:</strong><br/>"; 
Bo: while ($colors = mysqli_fetch_array($get_colors_res}) { 

Sa item color = $colors['item color']; 

D5: $display_block .= $item_color. "<br/>"; 

56: } 

Of: } 

5a: /ifree result 

59: mysSqli free result($get colors res); 

69: 

61: / iget sizes 

62: $get_sizes sql = "SELECT item size FROM store item size WHERE 

63: item_id = ".$safe_item_id." ORDER BY item_size"; 
64: $get_sizes res = mysqli query($mysqli, $get sizes sql) 

65: or die(mysqli_error{$mysqli)); 

66: 

67: if (mysqli_num_rows($get_sizes res} > 0) { 

68: ¢display block .= "<p><strong>Available Sizes:</strong><br/>"; 
69: while ($sizes = mysqli fetch array($get sizes res)) { 

70: $item size = $sizes['item_size']; 

ris $display block .= $item size."<br/>"; 

72: } 

rae } 

74: /ifree result 

To: mysqli_free_result($get_sizes_res}; 

TO: 

tf $display block .= "</div>"; 

78: } 


79: //close connection to MySQL 
80: mysqli _close($mysqli) ; 


Bai T> 

82: <!DOCTYPE html> 

83: <html> 

84: <head> 

85: <title>My Store</title> 

86: </head> 

87: <body> 

88: <?php echo $display block; ?> 
89: </body> 

90: </html> 


在 第 3 行 ， 进 行 了 数据 库 的 连接 ， 因 为 数据 库 中 的 信息 构成 了 这 个 


页 面 的 所 有 内 容 。 在 第 5 行 ， 开 始 了 $display_block 字 符 串 ， 其 中 市 有 一 
些 基 本 的 页 面 标题 信息 。 


第 11 行 到 第 13 行 创建 并 执行 了 一 个 得 询 ， 来 获取 分 类 和 商品 信息 。 
这 个 特定 的 查询 是 一 个 表 连 接 。 这 个 查询 只 是 根据 分 类 ID 来 连接 表 从 而 
获取 分 类 名 称 ， 而 不 是 从 一 个 表 选 取 了 商品 信息 然后 执行 一 个 二 次 查询 来 
FRAP RA PE o 


第 15 行 检查 结果 。 如 果 表 中 没有 匹配 的 商品 ， 一 条 消息 将 显示 给 用 
户 ， 并 且 脚 本 除 此 以 外 什么 也 不 做 。 然 而 ， 如 果 找 到 了 商品 信息 ， 脚 本 
继续 并 且 在 第 23 行 到 第 30 行 收集 信息 。 


在 第 34 行 到 第 35 行 ， 我 们 创建 了 上 所谓 的 面包 屑 路 径 (breadcrumb 
trail) 。 这 是 用 来 回 到 上 级 页 面 的 一 个 导航 。 上 述说 法 的 含义 是 “显示 出 
一 个 链接 以 便 我 们 能 够 回 到 分 类 ”。 这 个 脚本 中 的 分 类 ID 获取 目 主 奉 
询 ， 并 且 附 加 到 面包 居 路 径 中 的 链接 之 后 。 


在 第 36 行 到 第 39 行 ， 我 们 继续 添加 $display_block， 为 有 关 商 品 的 信 
居 建 立 一 个 表格 。 我 们 使 用 在 第 23 行 到 第 30 行 收集 的 值 来 创建 一 个 图 像 
链接 ， 显 示 出 说 明 以 及 价格 。 所 缺 的 就 是 颜色 和 大 小 ， 因 此 第 46 行 到 第 
57 行 选择 并 显示 出 和 这 一 商品 相关 的 任何 颜色 ， 并 且 第 62 行 到 第 73 行 收 
集 了 和 这 个 商品 相关 的 尺寸 。 


第 77 行 到 第 78 结 束 了 $display_block 字 符 串 和 主 if...else 语 句 ， 并 量 
由 于 脚本 没有 其 他 事情 需要 做 ， 它 显示 了 HTML (第 82 行 到 第 90 行 〉， 
包括 $display_block 字 符 串 的 值 。 图 22-3 显 示 了 当 从 hats 分 类 中 选择 棒球 
由 后 脚本 的 输出 。 当 然 ， 你 的 显示 可 能 和 我 的 不 同 ， 但 是 你 应 该 知道 其 


中 的 原因 。 


:=| 
| My Store 
€ © | © http://localhost/22/showitem.php7item_id=1 IKE 


My Store - Item Detail 


You are viewing: 


HATS > Baseball Hat 


Description: 
Fancy, low-profile baseball hat. 


Price: $12.00 





Available Colors: 
black 

blue 

red 


Available Sizes: 
One Size Fits All 
22-3 PESKIN TA rm W M 


Wie Bll a — “PS fal i NA EEA TERR QB, RATKA 
这 个 脚本 ， 以 便 可 以 把 商品 梁 加 到 一 个 购物 车。 


22DA Ne 


在 这 个 动手 实践 的 一 章 中 ， 我 们 应 用 了 基本 的 PHP 和 MySQL 知 识 来 
创建 一 个 商店 的 显示 。 我 们 学 习 了 如 何 创 建 数据 库 表 以 及 浏览 分 类 、 商 
品 列 表 和 单个 商品 的 脚本 。 


225 Q&A 


Q: 在 商品 明细 记录 中 ， 我 们 在 item_image 字 段 中 使 用 单个 的 
文件 名 ， 如 果 想 要 链接 到 不 同 的 服务 右上 的 商品， 该 怎么 
Tp? 


A: 可 以 在 item_image 字 段 输 入 一 个 URL， 只 要 我 们 定义 了 字段 来 
存储 URL 这 样 的 长 字符 串 。 


22.6 ”实践 练习 


这 个 实践 练习 设计 用 来 帮助 你 预测 可 能 的 问题 、 复 习 已 经 学 过 的 知 
识 ， 并 且 开 始 把 知识 用 于 实践 。 


1. 哪个 PHP 函 数 用 来 把 分 类 标题 字符 串 大 写 ? 

2. 为 什么 store item size 和 和 store item _color 表 没有 包含 任何 主键 或 
唯一 键 ? 

3. 为 什么 对 将 要 用 于 数据 库 奏 询 的 值 继续 使 用 


mysqli_real_escape_string()? 


1. strtoupper() 


2. RATEM WRIA AS HAAS AR. Al, 
item id 不 是 一 个 主键 或 唯一 键 。 另 外 ， 不 同 商 品 可 以 具有 相同 的 凑 色 或 
大 小 ， 因 此 item _color 和 和 item size 字段 一 定 不 是 主键 或 唯一 键 。 


3. 你 应 该 使 用 mysqli_real_escape_string() 来 确保 来 自用 户 的 值 可 以 
安全 地 使 用 ， 这 些 值 要 用 于 数据 库 的 查询 中 ， 不 论 你 是 要 创建 一 段 脚 
本 、 十 段 脚 本 或 一 百 段 脚本 ， 使 用 起 来 要 足够 安全 。 


BA Wel 


1. 通过 用 MySQL 执 行 自 己 的 查询 ， 再 创建 3 个 分 类 ， 每 个 分 类 中 
市 有 一 项 或 两 项 商品 。 


2. 为 你 存储 的 项 目 中 的 每 一 项 制作 一 些 图 像 〈 或 者 使 用 创作 共用 
许可 的 图 像 )， 并 且 ， 将 它们 放置 到 你 目 己 的 服务 器 上 的 一 个 图 像 目录 
中 。 对 showitem.php 做 一 些 必要 的 修改 ， 把 该 图 像 目 杂 添 加 到 生成 的 
<img/> 标 俭 的 文件 路 径 中 。 


第 23 划 ”创建 一 个 购物 车 机制 


在 本 章 中 ， 你 将 学 到 : 


e。 如 何 为 购物 车 和 在 线 商 店 的 结账 部 分 创建 一 个 关系 表 。 
e 如 何 创建 添加 和 删除 购物 车 商品 的 脚本 。 
。 处 理事 务 的 一 些 方法 ， 以 及 如 何 创 建 目 己 的 结账 流程 。 


在 这 个 动手 实验 的 一 草 中 ， 我 们 将 在 第 22 章 中 已 创建 的 商店 中 整合 
一 个 购物 车 机 制 和 结账 过 程 。 我 们 将 会 看 到 创建 相关 数据 库 表 的 方法 ， 
以 及 用 来 添加 和 删除 购物 车 商品 的 脚本 。 再 一 次 ， 本 章 使 用 的 例子 只 是 
展示 了 完成 这 一 任务 的 无 数 种 可 能 中 的 一 种 ， 并 且 它 只 是 一 个 工作 示例 
而 不 是 构建 在 线 商 店 的 权威 指南 。 


23.1 规划 和 创建 数据 库 表 


由 于 本 章 的 目标 是 为 用 户 提 供 选 择 和 订购 商品 的 一 种 方法 ， 所 以 可 
ae 象 一 下 这 些 表 是 什么 样 的 。 首 先 ， Se naan due 

。 除 了 cart 表 ， 我 们 还 需要 一 个 表 来 保存 订单 ， 要 一 个 表 来 保存 
作为 每 个 订单 的 一 部 分 的 所 购 商 上 HH o 


如 下 的 SQL 语句 用 来 创建 这 3 个 新 的 表 ， 从 store_shoppertrack 表 开 
始 。 当 用 刀 把 商品 添加 到 购物 车 的 时 候 ， 这 些 表 用 来 保存 它们 。 


提示 : 





用 来 定义 这 些 表 的 字段 长 度 是 随意 选取 的 ， 为 了 适合 多 种 可 能 的 输入 。 请 目 行 修改 这 些 
长 度 以 满足 你 目 己 的 特定 需求 。 


CREATE TABLE store shoppertrack ( 
id INT NOT NULL PRIMARY KEY AUTO INCREMENT, 
session_id VARCHAR (32), 
sel item_id INT, 
sel_item_qty SMALLINT, 
sel item _size VARCHAR(25), 
sel item color VARCHAR(25), 
date added DATETIME 


在 这 个 表 中 ， 唯 一 的 键 束 是 记录 的 id 字 上 段 。session_id 个 古 唯 一 的 ， 
售 则 ， 有 用户 只 能 从 商店 订购 一 件 商品 ， 这 不 是 做 生意 的 好 方法 。 


存储 在 session_id 字 段 中 的 值 代 表 用 户 ， 它 和 分 配给 用 户 的 PHP 会 话 
ID 的 值 相 匹配 。selL_* 字 段 保 存 了 用 户 的 选择 : 所 选 的 商品 、 所 选 商 品 的 
数量 以 及 所 选择 的 商品 的 凑 色 和 大 小 。 最 后 ， 古 一 个 date_added 字 段 。 


很 多 时 候 ， 用 户 把 商品 放 入 到 购物 和 车 中 但 却 没 有 去 结账 。 这 一 动作 实际 
上 在 记录 表 中 留 下 了 散乱 的 商品 ， 你 可 能 想 归 定期 清理 。 例 如 ， 你 可 能 
想 删 际 一 周 之 前 的 所 有 购物 车 中 的 商品 ， 这 不是 date_added 了 字段 的 用 武 
之 地 。 


下 一 个 表 你 存 了 订购 信息 。 


CREATE TABLE store_orders ( 
id INT NOT NULL PRIMARY KEY AUTO INCREMENT, 
order_date DATETIME, 
order_name VARCHAR (100), 
order_address VARCHAR {255}, 
order_city VARCHAR (50), 
order state CHAR(2), 
order zip VARCHAR(10), 
order tel VARCHAR(25}, 
order email VARCHAR(1@9), 
item total FLOAT(6,2}, 
shipping total FLOAT(6,2), 
authorization VARCHAR (590), 
Status ENUM('processed', ‘pending’ ) 


store_orders 表 中 的 唯一 的 键 字 段 是 jd。 为 了 本 草 的 简短 起 见 ， 假 设 
用 尸 的 订单 和 送 仙 地 址 是 相同 的 ， 并 且 这 个 商店 只 是 问 美 国 以 内 的 地 址 
销售 商品 。 为 大 货 地 址 信息 添加 另 一 组 字段 也 是 很 容易 的 ， 如 有 果 你 愿意 
这 么 做 的 话 。 如 采 这 是 一 个 现实 的 例子 ， 你 肯定 想 要 添加 字段 ; A, 
对 于 那些 想 要 购买 东西 作为 礼物 的 购物 着 来 说 ， 你 将 会 失去 做 生意 的 机 
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这 么 做 。 相 反 ， 这 个 表 是 根据 实时 的 思路 来 处 理 信用 卡 。 你 将 会 在 本 章 
最 后 学 习 一 些 事务 选项 。 最 终 你 和 存 每 个 订单 的 每 行商 品 的 表 古 


store orders items. 


CREATE TABLE store_orders_items ( 
id INT NOT NULL PRIMARY KEY AUTO_INCREMENT, 
order Id INT, 
sel item id INT, 
sel item dty SMALLINT, 
sel_item_size VARCHAR(25}, 
sel _ item color VARCHAR({25), 
sel_item_price FLOAT(6,2) 


sel _* 字 段 看 上 去 似 兽 相似 ， 除 了 sel_item_price 之 外 ， 它 们 和 
store_shoppertrack 表 中 的 字段 是 相同 的 。 主 键 是 id 字段 ， 并 且 order id 字 
段 用 来 把 每 行商 品 和 store_orders 中 相应 的 记录 联系 起 来 。 


这 里 也 包含 了 sel_item_price 字 段 ， 而 不 是 简单 地 关联 到 商品 记录 ， 
因为 我 们 可 能 有 机 会 要 在 商品 记录 中 修改 价格 。 如 果 我 们 在 商品 记录 中 
修改 价格 ， 并 且 把 已 销售 的 每 个 商品 关联 到 当前 的 新 价格 ， 我 们 的 每 个 
己 销 售 丙 品 的 价格 将 不 会 反映 用 户 实际 的 文 付 。 现 在 ， 所 有 的 表 都 已 经 
ERT, FERITE a) SF as TI Fe m o 


23.2 FOU ES SA 


在 本 节 中 ， 我 们 将 修改 第 22 章 中 的 showitem.php 脚 本 。 有 目标 是 把 商 
品 信 息 页 面 转 换 为 融 有 一 个 用 来 选择 颜色 、 大 小 和 数量 的 表单 的 商品 信 
ER H o 


在 最 初 的 脚本 中 ， 在 第 2 行 之 前 插入 如 下 内 容 。 
session start(}; 
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下 一 个 修改 在 最 初 脚 本 的 第 39 行 才 出 现 ， 因 此 ， 我 们 的 程序 清单 23.1 从 
那里 开始 。 


程序 清单 23.1 showitem.php 中 的 新 代码 


39: 
Aa: 
Ar 
Ae: 
433 
44: 
45; 
46: 
473 
45: 
Ag: 
5@:! 
5I: 
ae 
Ba: 
54: 
B5: 
56: 
57- 
58: 
59: 
60: 
61: 
G2! 
63: 
64: 
65: 
66: 
67: 
68: 
69: 
f@: 
7i: 
T2: 
Ta: 
rf: 
oe 
46; 
rite 
483 
79: 
80: 
81: 
62 | 
83: 


<p><strang>Price:</strong> \$$item price</p> 
<form method="post" action="addtocart.php"> 


END _OF_TEXT; 


/ifree result 
mysqli free result($get item res}; 


{iget colors 

$get_colors sql = "SELECT item_color FROM store item color WHERE 
item id = '".$safe_ item _id."' ORDER BY item color"; 

$get colors res = mysqli query($mysgli, $get colors sql} 
or die(mysqli_error($mysqli)}; 


if (mysgqli_num_rows($get_colors res} > Ø) { 
$display block .= "<p><label for=\"sel item color\"> 
Available Colors:</label><br/> 
<select id=\"sel item colonr name=\"sel item _color\'>' | 


while ($colors = mysqli fetch array($get colors res}) { 
$item color = $colors['item_color']; 
display block .= “<option value=\"".$item color.'\">". 
$item_color."</option>"; 
} 
$display block .= "</select></p>"; 
} 


//tree result 
mysqli free result($get colors res}; 


{iget sizes 
$get_sizes sql = "SELECT item_size FROM store item size WHERE 
item id = ".$safe item id." ORDER BY item size"; 
$get_sizes res = mysqli query($mysqli, $get sizes sql} 
or die(mysgli error({$mysqli)}; 


if (mysqli num rows($get sizes res} > @) { 
$display block .= "<p><label for=\"sel item size\"> 
AvVallable Sizes:</label><br/> 
<select id=\"sel item size\" name=\"sel item size\">"; 


while ($sizes = mysqli fetch array($get sizes res}} { 
$item size = $sizes['item size']; 
$display block .= "<option value=\"".$item size."\">". 
$item size."</option=" ; 


84: } 


85: 

86: Gdisplay block .= "</select></p>"; 

87; 

88: //free result 

89: mysqli free result{$get sizes res); 

90: 

91: display block .= " 

92} <p><label Tor=\"sel_ item_qty\">Select Quantity:</label> 
93: <select id=\"sel item qty\" name=\"sel item qty\">"; 
94; 

95: for($i=1; $i<11; $i++) { 

96: $display_ block .= "<option value=\"".$1."\">".$i."</option>"; 
97: } 

98: 

99: display block .= <<<END OF TEXT 


100: </select></p> 

101: <input type="hidden" name="sel item_id" 
102: value="$ GET[item id]" /> 
183: <button type="submit" name= submit value="submit"> 
104: Add to Cart</button> 

105: </form> 

106: </div> 

107: END OF TEXT; 

108: } 

199: 

110: /fclose connection to MySQL 
111: mysqli close($mysq1i} ; 

Tea: 7> 

113: <!DOCTYPE html> 

114: <html> 

115: <head> 

116: <title>My Store</title> 

117: <style type="text/css"> 

118: label {font-weight: bold; } 
119: </style> 

120: </head> 

121: <body> 

122: <?php echo $display block; 7> 
123: </body> 

124: </html> 


第 一 个 改变 是 在 新 代码 的 第 40 行 ，$display_block 字 人 符 串 奶 加 了 包含 
开始 的 <form> 元 聚 。 表 时 的 动作 就是 一 个 名 为 addtocart.php 的 脚本 ， 我 
们 将 在 下 一 区 创建 这 个 脚本 。 


下 一 个 改变 出 现在 第 55 行 ，$display_block 字 符 串 追加 了 包含 
<Sselect> 元 叉 的 开始 标签 ， 其 名 字 为 sel item color (“Available Colors” 现 
在 由 第 53 行 到 第 54 行 的 <label> 标 签 包 围 了 起 来 ， 因 为 它 标 记 一 个 表单 元 
A) 。 在 第 58 行 到 第 60 行 ， 闫 色 放 入 到 <option> 元 系 中 供用 户 从 中 选 
择 ， 而 不 是 直接 显示 在 屏 科 上。 第 62 行 结束 <select> 元 系 。 


对 于 了 商品 大 小 进行 了 同样 类 似 的 修改 。 第 75 行 到 第 77 行 ， 
$display_block7¥ BiB IN S E444 Nsel_item_sizeH)<select>7t#. 380 
行 到 第 82 行 把 颜色 写 入 到 <option> 元 素 ， 并 且 第 86 行 结束 了 <select> 元 
Bo 


第 91 行 到 第 97 行 是 新 增添 到 脚本 中 的 。 这 些 代 码 创 建 了 一 个 名 为 
sel_item_qty 的 <select> 元 系 ， 用 于 用 户 选择 购买 多 少 商 品 。 第 100 行 结 
了 这 个 <select> 元 际 ， 并 且 第 101 行 到 第 102 行 同 item_id 球 加 了 一 个 隐 蔚 
字段 。 第 103 行 到 第 104 行 添加 了 一 个 提交 投 钮 ， 并 且 第 105 行 结束 了 表 
单 。 我 们 在 第 111 行 关闭 了 到 MySQL 的 连接 ， 剩 下 的 代码 和 最 初 的 脚本 
EEIT AEs 
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1 所 示 的 结果 ， 它 反映 出 了 表 早 元 系 的 增加 。 


Wy Store 


= C | © http://localhost/23/showitem.php7item_id=1 Ki A ® 


My Store - Item Detail 


You are viewing: 


HATS > Baseball Hat 


Description: 
_ Fancy, low-profile baseball hat. 


Price: $12.00 





Available Colors: 
black || 


Available Sizes: 
One Size Fits All |æ] 


Select Quantity: 4 [=| 


图 23-1 新 的 棱 球 帽 商 品 页 面 


下 一 步 是 创建 addtocart.php 脚 本， 以 全 刚刚 创建 的 表单 能 够 真正 地 
做 些 事情 。 


23.2.1 把 项 目 添加 到 购物 车 


addtocart.php 脚 本 只 是 把 信息 写 入 到 store_shoppertrack 表 并 且 把 用 户 
重 定 同 到 浏 虎 购物 车 。 我 们 将 首先 在 程序 清单 23.2 中 创建 addtocart.php 
脚本 ， 然 后 处 理 showcart.php 脚 本 。 


程序 清单 23.2 addtocart.php 脚 本 


<?php 
session start({); 


if (isset($ POST['sel item_id']}} { 


oa wrt oar wh — 


f/fconnect to database 


$mysqli = mysqli_connect("localhost", “joeuser", "Somepass", "testDB"); 


/fcreate safe values for use 

$safe sel item_id = mysqli real escape string($mysqli, 
$ POST['sel item id']}; 

$sate sel item_qty = mysqli_real_ escape string(S$mysqli, 
$ POST['sel item qty']); 

$safe sel item size = mysqli_real_escape_ string(Smysqli, 
$ POST['sel item size']); 

$safe sel item_color = mysqli real escape string{$mysqli, 
$ POST['sel item color']}; 


fivalidate item and get title and price 
$get_iteminfo_ sql = "SELECT item title FROM store_items WHERE 
id = '".$safe sel item id."'"; 
$S$get_iteminfo_ res = mysqli _query($mysqli, $get_iteminfo_ sql) 
or die(mysqli error($mysqli}); 


if {mysqli_num_rows($get_iteminfo_res) < 1) { 


//free result 
mysqli free_result($get_iteminfo_res} ; 


/ ¿close connection to MySQL 
mysqli close($mysqli) ; 


/finvalid id, send away 
header( "Location: seestore.php"}; 
exit; 
} else { 
{iget info 


while ($item info = mysqli _fetch_array($get_iteminfo_res)) { 


$item title = stripslashes($item info['item title']); 


} 


:ifree result 
mysqli free result($get_iteminfo_ res}; 


/fadd info to cart table 
$addtocart sql = "INSERT INTO store _shoppertrack 
(session id, sel item id, sel item qty, 


sel item_size, sel _item_color, date_addea) 


VALUES ({'".$ COOKIE['PHPSESSID']."', 
'" $safe_ sel item id."', 
",$safe_sel_item_qty.”’, 
",$safe sel item size."', 


.$safe sel item _color."', now(}}"; 
$addtocart res = mysqli query($mysqli, $addtocart_ sql) 
or die(mysqli_error($mysqli)); 


ats //close connection to MyS@L 


58: mysqli close ($mysqli)}; 

59: 

60: //redirect to showcart page 
61: header("Location: showcart.php"); 
62: exit; 

63: } 

64: 

65: } else { 

66: //send them somewhere else 

67: header("Location: seestore.php' ) ; 
68: exit; 

69: } 

70: ?> 


第 2 行 继 续 用 户 会 话 ， 这 很 午 要 ， 因 为 我 们 需要 捕获 用 户 的 会 话 ID 
来 写 入 到 store_ shoppertrack 表 。 


在 第 4 行 ， 这 个 脚本 验证 了 出 现在 $_ POST['sel item_id] 中 的 一 个 
值 ， 意 味 着 当 提 交 了 相应 的 表单 之 后 ， 用 户 来 到 这 一 脚本 。 如 果 没 有 
值 ， 脚 本 跳 到 第 51 行 并 且 在 第 53 行 发 送 给 用 户 ， 而 这 束 是 脚本 所 作 的 事 
情 。 

然而 ， 如 果 $_POST['sel item_id] 中 有 一 个 值 ， 第 6 行将 会 进行 数据 
库 连 接 ， 第 9 行 到 第 16 行 会 创建 表单 信息 的 安全 厂 本 ， 第 17 行 开始 将 执 
行 一 个 查询 来 验证 选择 的 商品 ID 是 一 个 有 效 的 商品 。 第 19 行 到 第 22 行 创 
建 并 执行 了 一 个 SQL 但 询 来 收集 选择 的 商品 的 标题 。 第 24 行 检查 一 个 结 
果 ， 如 果 没 有 结果 ， 用 户 在 第 33 行 再 次 重 定 同 ， 因 为 障 品 选择 是 无 效 
的 。 


如 果 了 商品 选择 有 效 ， 脚 本 继续 到 第 38 行 并 且 从 结果 集中 提取 值 。 这 
个 脚本 现在 有 是 够 的 信息 来 癌 store_shoppertrack 表 添加 商品 选择 ， 这 在 
第 45 行 到 第 54 行 完成 。 


当 查 询 执 行 完 ， 用 户 重 定 问 到 showcart.php， 其 中 包含 所 有 的 购物 
车 商品。 我 们 在 下 一 节 中 创建 这 个 脚本 。 
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23.3 给 出 了 showcart.php 的 代码 。 


程序 清单 23.3 ”showcart.php 肢 本 


<?php 
session start{); 


P O 一 


¿iconnect to database 


$mysqli 


= mysqli connect("localhost", “jceuser", "somepass", “testDB"); 


$display block = "<h1>Your Shopping Cart</h1i>"; 


{icheck for cart items based on user session id 
$get cart sql = “SELECT st.id, si.item title, si.item price, 


$get_cart_res = mysqli query($mysqli, $get_cart_sql} 


st.sel item qty, st.sel item_s176¢, 


st.sel item color FROM 


store shoppertrack AS st LEFT JOIN store items AS si ON 
si.id = st.sel_item_id WHERE session_id = 


”中 COOKIE[ PHPSESSID ]- 


or die(mysqli_error($mysqli)}; 


if (mysqli_num_rows($get_cart_res} < 1} { 
{iprint message 

$display block .= "<p>You have no items in your cart. 
Please <a href=\"seestore,.php\">continue to shop</a>!</p>"; 
} else { 


/ige 
$dis 
<tab 
<tr> 
<th> 
<th> 
<th> 
<th> 
<th> 
<th> 


t info and build cart display 
play block .= <<<END OF TEXT 
le> 


Title</th> 
Size</th> 
Color</th> 
Price</th> 
Qty</th> 

Total Price</th> 


<th=Action</th> 
</tr> 
END OF TEXT; 


whil 


e ($cart_info = mysqli fetch array($get cart _res)) 1 


$id = Scart_info['id']; 


Sitem title = stripslashes($cart_info['item_title’)}; 


$item price = $cart_info[ item price ]; 
$item qty = $cart_info['sel item qty']; 
$item color = $cart_info[ 'sel_item_color' ] ; 
$item size = $cart_info['sel_item_size']; 
$total price = sprintf("%.@2T", $item price 


$display block .= <<<END OF TEXT; 
<tr> 

<td>$item title <br></td> 
<td>$item_size <br></td> 
<td>$item color <br></td> 

<td>\$ $item price <br></td> 
<td>$item_qty <br></td> 

<td>\$ $total price</td> 


* $item qty); 


<td><a href="removefromcart.php?id=$id">remove</a></td> 


</tr> 


END OF TEXT; 


} 


+ 


$display block .= "</table>"; 


//free result 
mysqli free result($get cart res); 


63: //close connection to MyS@L 
64: mysqli close($mysqli} ; 

65: 7> 

66: <!DOCTYPE html> 

67: <html> 

68: <head> 

69: <titleoMy Store</title> 

7@: <style type="text/css"> 


Pls table { 

Gee border: 1px solid black; 
Tas border-collapse; collapse; 
T4: } 

Fas th { 

76: border: 1px solid black; 
77 : padding: 6px; 

r8: font-weight: bold; 

79: background: #ccc; 

80: text-align: center; 

81: } 

82: td { 

83: border: ipx solid black; 
84: padding: 6px; 

85: vertical-align: top; 

86: text-align: center; 

87 : } 

88: </style> 

89: </head> 

90: <body> 

91: <?php echo $display block; ?> 
92: </body> 

93: </html> 


第 2 行 继续 用 户 会 话 ， 这 很 重要 ， 因 为 我 们 需要 用 户 会 话 ID 来 匹配 
store_shoppertrack 表 中 的 记录 。 第 5 行进 行 数据 库 连 接 ， 第 7 行 开始 了 
$display_block 字 符 串 ， 其 中 市 有 一 个 页 面 的 标题 。 


第 10 行 到 第 14 行 表示 一 个 连接 僵 询 ， 其 中 ， 获 取 了 用 户 的 保存 疝 
品 。id、sel_item_qty、sel_item_size 和 sel_item_color 字 段 都 获取 目 
store_shoppertrack ; 根据 匹配 来 目 store_shoppertrack 的 信息 ，item_title 
和 item_price 字 段 获取 上 自 store_items 表 。 换 句 话 说 ， 对 于 选择 的 商品 ， 


Baseball Hat CHEKIB) 显示 为 标题 ， 而 不 是 显示 2。 第 15 行 到 第 16 行 执 
行 了 这 个 查询 ， 第 18 行 检查 了 结果 。 


如 果 没 有 结果 ， 用 户 在 store_shoppertrack 表 中 融 没 有 商品 。 一 条 消 
思 写 入 到 $display_block 字 人 符 串 ， 脚 本 退出 并 蛇 示 消息 。 


如 果 有 真正 的 结果 ， 第 24 行 到 第 34 行 创建 了 一 个 HTML 表 格 的 开 
头 ， 禹 有 为 购物 车 中 的 所 有 “也 可 能 是 某 些 ) 信息 而 定义 的 列 。 第 37 行 
开始 了 一 个 while 循 坏 ， 从 store_shoppertrack 中 提取 每 个 商品 ， 并 且 这 个 
人 循环 持续 到 第 56 行 ， 把 信息 输出 到 相应 的 表格 单元 格 中 。 


在 第 54 行 ， 我 们 看 到 创建 了 一 个 到 删除 商品 脚本 的 链接 ， 我 们 将 在 
下 一 节 中 创建 这 个 脚本 。 第 58 行 结束 了 这 个 表格 ， 访 脚本 在 第 66 行 到 第 
93 行 结束 并 且 问 屏幕 显示 HTML， 包 括 了 表格 标题 和 其 他 单元 格 所 使 用 
HY) — HERE SUZ 


现在 ， 我 们 回 到 一 个 商品 页 面 并 且 把 商品 添加 到 购物 车 。 在 商品 与 
入 到 store_shoppertrack 表 之 后 ， 我 们 应 该 重 定 加 到 Showcart.php 页 面 ， 并 
上 且 新 选择 的 商品 也 应 该 显示 。 图 23-2 显 示 了 添加 一 些 商品 之 后 的 购物 
ae 
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图 23-2 ”添加 a 到 购物 车 的 商品 


下 一 步 就 是 创建 ramovefromcart.php 脚 本 。 


23.2.3 ”从 购物 车 中 删除 项 目 
removefromcart.php 脚 本 很 得 小， 因为 它 所 有 做 的 就 是 执行 一 个 得 询 
并 且 把 用 户 重 定 同 到 男 一 个 脚本 。 不 可 避免 的 ， 用 户 想 要 从 他 的 购物 车 
中 删除 项 目 ， 并 且 这 个 脚本 也 恰恰 是 允许 他 做 到 这 一 点 的 。 程 序 清 单 
23.4 显 示 了 完整 的 脚本 。 
程序 清单 23.4 _ removefromcart.php 脚 本 


1: <?php 

2: Session. start}; 

i 

4; if (isset($ GET['id'])}) { 

a //connect to database 

6: $mysqli = mysqli connect("localhost", "joeuser”, "Somepass", "testDB"); 
ya 

8: //create safe values for use 

9: $safe id = mysqli_real_escape string($mysqli, $ GET[ 29 ] ) 
10: 

Lh $delete item_sql = "DELETE FROM store_shoppertrack WHERE 
12: id = '".$safe_ id."' and session id = 
3: "we GOOKLET PHPSESSID =: 

las $delete item res = mysqli query{$mysoli, $delete item sql} 
TO or die{mysqli error($mysql1i)); 

O 

12 //close connection to MySQL 

18: mysqli close($mysqli) ; 

19: 

20: //redirect to showcart page 

2 header{"Location: showcart.php"); 

oe. exit; 

23: } else { 

24: //send them somewhere else 

25: header({"Location: seestore.pnp"); 

26: exit; 

27: } 

28: ?> 


第 2 行 继续 用 户 会 话 ， 因 为 我 们 需要 使 用 用 户 会 话 ID 来 和 
store_shoppertrack 表 中 的 记录 相 逻 配 。 第 4 行 检 查 了 $_GET['id'"] 中 的 值 。 
如 采 $_GET[id] 中 不 存在 一 个 值 ， 用 户 不 能 单 击 来 自 目 己 的 购物 车 的 链 
接 ， 从 而 ， 在 第 22 行 发 送 到 别处 。 


如 果 $_GETFid] 中 存在 一 个 值 ， 第 6 行进 行 数据 库 连 接 ， 第 9 行 创 建 
该 变量 的 一 个 数据 库 安 全 版 本 ， 在 第 14 行 到 第 15 行 执行 第 11 行 到 第 13 行 
的 那个 SQL 得 询 ， 并 且 用 户 在 第 21 行 重 定 同 到 Showcart.php script， 其 
中 ， 补 删除 的 商品 将 不 再 显示 。 答 试看 看 。 


23.33 ” 文 付 方法 和 结账 过 程 


当 到 了 为 购物 车 中 购买 的 商品 而 付款 的 时 候 ， 和 存在 儿 种 两 业 方法 。 
对 你 来 说 , “正确 的 ”方法 取决 于 你 的 生 音 ， 通 过 银行 的 经 两 账户 通 第 要 
求 你 有 一 个 营业 执照 、 一 个 分 销 许可 以 及 其 他 的 证 书 来 证 明 你 做 的 是 合 
法 生 章 。 如 来 你 只 是 想 要 买 挥 一 些 丙 品 的 个 人 ， 可 能 丰硕 望 经历 所 有 这 
些 文书 工作 。 然 而 ， 你 仍然 有 选择 。 


不 管 选 择 什么 样 的 支付 方式 ， 有 一 件 事情 是 肯定 的 ， 如 果 要 通过 
Web 传 递 信用 卡 信息 ， 必 须 通过 一 个 SSL 连 接 来 实现 。 第 30 章 介绍 了 如 
何 获取 一 个 SSL 认 证 并 且 将 其 安装 在 你 的 系统 上 。 我 们 不 一 定 要 在 用 户 
的 整个 购物 过 程 中 都 使 用 安全 连接 ， 只 是 从 捕获 敏感 信息 的 时 候 ， 例 如 
结账 页 面 ， 开 始 使 用 这 一 连接 就 行 了 。 


23.3.1 创建 结账 页 面 


现在 ， 你 应 该 已 经 很 好 地 掌握 了 一 个 简单 页 面 的 创建 。 在 本 半 的 开 
始 ， 创 建 了 带 有 一 些 字 段 的 store_orders 表 ， 这 些 字 段 作 为 页 面 的 一 个 指 


e Order name 
e Order address 


e order_city 


order_ state 


order_zip 


order_tel 


e order email 


Boob, BUEN Aa BA aR SAS. ROAD ea BY 
BFF Bl. Ib “Mase Be ce EA Pei me DT AZ APE 
物 车 的 内 容 ， 以 便 顾 客 能 够 记得 他 正在 付 葡 以 及 这 笔 订 单 大 概要 化 多 少 
钱 。 也 是 在 结账 过 程 的 这 个 时 刻 ， 我 们 提供 了 可 能 拥有 的 任何 达 贷 ; 
项 。 送 货 和 销售 税 将 在 此 过 程 的 下 一 个 步骤 中 计算 。 


从 操 击 表 早 上 的 提交 按钮 开始 ， 结 账 过 程 取 决 于 你 所 使 用 的 支付 方 
法 。 下 一 市 将 介绍 基本 的 步 又 并 且 提 供 有 关 文 付 过 程 的 各 种 方法 的 建 


议 。 
23.3.2 ”执行 结账 操作 


如 采 我 们 已 经 通过 银行 获得 了 一 个 经 商 账户 ， 可 以 利用 像 PayPal 的 
PayFlo Pro 或 Authorize. Net 的 文 付 网 天 服务 这 样 的 实时 文 付 服务 。 要 了 
解 天 于 这 些 服务 的 更 多 信息 ， 请 分 别 访问 http://www.authorize.net 和 


https://www.paypal.com/webapps/mpp/merchant 。 


PHP 并 没有 包含 内 建 的 函数 来 支持 把 访问 叶 问 这 些 文 付 网 天 ， 但 
是 ， 妆 你 考虑 使 用 这 些 类 型 的 商户 服务 的 时 候 ， 可 以 下 载 到 可 用 于 目 己 
的 应 用 程序 的 脚本 ， 或 者 会 得 到 供应 用 程序 开 友 者 使 用 的 一 个 API 的 信 


FA 
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PayPal 和 Authorize.Net 是 已 有 的 儿 个 供 商 户 使 用 的 交易 处 理 网 关中 
的 两 个 ， 我 在 这 里 介绍 它们 ， 是 因为 我 个 人 从 这 两 个 网 关 服 务 诞 生起 就 
使 用 它们 。 你 的 银行 通常 会 提供 一 个 建议 你 使 用 的 商户 的 列表 。 如 果 你 
不 知道 该 从 银行 提供 的 商户 列 表 中 选择 哪个 ， 确 保 彻 夺 地 研究 你 选择 的 


商户 ， 以 避免 资金 延期 并 且 确 保 你 能 够 得 到 最 好 的 服务 。 
在 选择 了 一 个 事务 处 理 右 之 后 ， 结 账 脚本 应 该 加 从 如 下 的 步 又 。 


1. 计算 商品 、 税 和 送 货 费 的 总 和 。 这 求 得 一 个 要 从 信用 卡 扣 款 的 


2. 对 总 数 执行 信用 卡 授 权 。 


3. 从 卡 处 理 程序 得 到 一 个 成 功 的 或 失败 的 啊 应 。 如 末 啊 应 失败 ， 
问 用 户 显 示 一 条 消 轧 ， 并 且 事 务 结束 。 如 琳 吧 应 成 功 ， 继 续 步 又 4。 


4. 把 基本 的 订购 信息 写 入 到 像 store_orders 这 样 的 一 个 表 中 ， 包 括 
我 们 将 接收 到 成 功 授 权 的 授权 代码 。 使 用 mysgql_insert_id0) 获 取 这 条 记 杂 
的 id 值 。 


5. 对 于 购物 车 中 和 这 个 用 户 相 关 的 每 件 商 品 ， 癌 
store_orders_itemmap 中 插入 一 条 记录 。 每 条 记录 都 将 引用 在 上 一 个 步骤 
中 收集 来 的 id (作为 order_ id) 。 


6. 删除 用 户 的 购物 车 中 的 商品 。 


7. 在 屏幕 上 显示 带 有 授权 代码 而 不 是 信用 卡 信息 的 订单 ， 以 便 用 
户 可 以 打印 它 并 保存 为 收据 。 我 们 也 可 以 通过 Email 将 这 一 信息 发 送 给 
AAR. 
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又 ， 并 且 我 们 将 在 本 书 整个 过 程 中 用 到 ， 并 且 ， 没 有 理由 让 它们 变 得 更 
加 复杂 。 


23.4 ”小 结 


在 这 个 动手 实验 的 一 革 中 ， 我 们 应 用 基本 的 PHP 和 MySQL 知 识 把 一 
个 购物 车 整合 到 第 22 半 的 两 店 中 。 包 括 数 据 库 表 的 创建 ， 修 改 商 品 细节 
页 面 ， 以 及 洪 加 和 删除 购物 车 商品 的 狐 脚 本 。 


23.9 Q&A 


Q: 当 用 户 将 商品 加 入 到 自己 的 购物 车 的 时 候 ， 他 们 如 何 确 
定 该 项 商品 有 库存 ? 


A: 如 果 修 改 了 store_items 表 ， 而 该 表 包 食 了 表示 当前 存 仙 量 的 一 

个 字段 ， 并 且 ， 当 用 户 完 成 了 结账 过 程 ， 和 存货 量 减 去 用 户 订 购 的 书目 ， 

然后 ， 在 showitem.php 脚 本 中 ， 我 们 可 以 生成 一 个 下 拉 列 表 ， 其 中 上 其 有 
该 商品 现 有 库存 可 供 选 择 的 最 大 数目 。 当 然 ， 如 有 果 操 作 库 存 汇 总 的 上 干 
种 商品 ， 如 条 你 的 下 拉 选 择 器 只 多 许 每 次 购买 10 个 商品 的 话 ， 那 没什么 
天 系 。 然 而 ， 为 了 实现 更 好 的 用 户 体 验 ， 你 可 能 需要 让 用 户 能 够 回 购物 
匣 谎 加 尽 可 能 多 的 商品 ， 在 这 种 情况 下 ， 在 完成 添加 到 购物 年 操作 之 

前 ， 还 需要 这 加 一 个 库存 检查 功能 ， 并 且 ， 不 允许 添加 到 购物 车 中 的 数 
量 比 库存 中 的 数量 还 大 。 


23.6 ”实践 练习 


这 个 实践 练习 设计 用 来 帮助 你 预测 可 能 的 问题 、 复 习 已 经 学 过 的 知 
识 ， 并 且 开 始 把 知识 用 于 实践 。 


问答 题 


1. 当 从 购物 车 中 删除 一 个 商品 时 ， 为 什么 要 查询 根据 该 记录 验证 
用 户 的 会 话 ID? 


2. 汪 加 a 到 购物 车 的 时 候 ， 没 有 把 价格 存储 在 一 个 隐藏 字段 中 的 原 


3. 如 果 圾 要 满足 壕 代 地址 和 付 球 地 址 不 同 的 需求 ， 填 要 对 数据 库 
和 表单 做 些 什 么 ? 


oy 


eA 
1. 用 户 应 该 只 能 够 删除 他 自己 的 购物 车 中 的 商品 。 


2. 如 采 我 们 把 价格 存储 在 一 个 隐 羧 字段 中 ， 一 个 恶意 的 用 刀 能 够 
在 这 个 值 及 布 到 表单 之 前 修改 它 ， 从 而 ， 可 以 在 store_shoppertrack 表 中 
写 入 他 想 要 的 价格 ， 而 不 是 实际 的 价格 。 


.在 store_orders 表 中 修改 已 有 的 与 地 址 相关 的 字段 ， 以 清晰 地 标 


we 起 地 址 还 是 付 球 地 址 ， 人 然后， 在 表 中 复制 这 些 字 上 段 的 组 合 
(给 它们 一 个 名 字 ， 标 明和 它们 是 送 货 地 址 还 是 付款 地 址 ) ， 在 表单 和 最 


终 的 INSERT 语 句 中 也 这 么 做 。 
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的 任何 其 他 页 面 ) 的 链接 。 


2. 在 创建 了 “结账 ?表单 之 后 ， 将 订 贡 信息 和 列表 项 保存 到 本 和 章 开 
始 处 创建 的 数据 库 表 中 。 此 时 移 不 必 担 心经 商 账户 或 文 付 过 程 。 这 个 过 
旦 只 是 ， 人 允许 用 户 将 商品 保存 到 其 购物 车 中 ， 给 你 东 个 虚拟 的 信用 卡 和 
地 址 信息 来 结账 ， 然 后 ， 将 基本 的 订货 信息 存储 到 store_orders 表 中 ， 将 
订购 的 商品 存储 到 Store_orders_items 和 中 。 


第 24 章 ”创建 一 个 向 单 的 日 历 


ERER, MÄ FI: 


。 如 何 构建 一 个 简单 的 日 历 脚 本 。 
。 如 何在 日 历 中 查看 和 添加 事件 。 
。 如 何 构建 一 个 类 库 来 产生 HTML 表 单 中 的 日 期 下 拉 列 表 。 


本 草 将 把 我 们 到 目前 为 止 已 经 学 习 a 到 的 有 关 PHP 语 言 和 构建 应 用 程 
序 的 内 容 揉 合 到 一 起 。 在 本 重 中 ， 我 们 将 在 创建 一 个 简单 日 历 的 青 景 
继续 学 习 。 


24.1 构建 一 个 人 徐 单 的 显示 日 历 


我 们 将 使 用 在 第 10 章 中 学 习 的 日 斯 和 时 间 孔 数 来 构建 一 个 日 历 ， 它 
将 显示 1980 年 到 2010 年 之 间 的 任何 月 份 的 日 期 (这 是 随机 选择 的 年 份 ， 
没有 任何 是 义 ， 如 果 你 愿意 ， 可 以 让 目 己 的 日 历 范 围 从 1990 年 到 2005 
Œ) 。 用 户 将 能 够 通过 下 拉 沫 单 选 择 年 份 和 月 份 ， 并 且 选 择 的 月 份 的 日 
期 投 照 星期 几 来 组 织 排 列 。 


在 这 个 脚本 中 ， 我 们 将 使 用 两 个 变量 ， 一 个 用 于 月 份 ， 一 个 用 于 年 
份 ， 这 两 个 变量 将 通过 用 户 和 输入 皖 供 。 这 些 信息 片断 将 用 来 根据 所 选 月 
份 的 第 一 天 来 构建 一 个 时 间 戳 。 如 采用 户 输入 无 效 或 者 巷 失 ， 默 认 信 将 
会 是 当前 月 份 的 第 一 天 。 


24.1.1 检查 用 户 输 入 


当 用 尸 第 一 人 次 访问 这 个 日 历 应 用 程序 的 时 候 ， 还 没有 提交 任何 信 
晨 。 因 此 ， 我 们 必须 确 你 脚本 能 够 处 理 这 样 的 事实 ， 束 是 月 和 年 的 变量 
可 能 还 没有 定义 。 我 们 使 用 issetO 函 数 来 做 到 这 一 点 ， 如 采 传 递 给 它 的 
变量 还 没有 定义 ， 该 图 数 将 返回 false。 然 和 而， 让 我 们 使 用 checkdateO 函 
数 来 蔡 代 它 ， 这 个 函数 不 仅 得 看 变量 是 售 存 在 ， 而 且 还 对 它 做 一 些 有 意 
义 的 事情 ， 即 验证 它 是 一 个 日 期 。 程 序 清 单 24.1 给 出 了 一 个 代码 段 ， 它 
检 奏 来 和 目 一 个 表单 的 nonth 和 year 变 量 ， 并 根据 它们 构建 一 个 时 间 答 。 


程序 清单 24.1 检查 来 目 日 历 脚 本 的 用 户 输入 


1: <?ph 

2: if ((!isset($_POST['month'])) || (!isset($_POST['year'])})} { 
3a $nowArray = getdate(}; 

4: $month = $nowArray['mon' ]; 

53 $year = $nowArray['year']; 

6: } else { 

7: $month = $_POST['month']; 

8: $year = $ POST['year']; 

9: 1} 


1@: $start = mktime (12, @, @, $month, 1, $year); 
11: $firstDayArray = getdate($start}: 
12: 7> 


程序 清单 24.1 只 古 一 个 较 大 的 脚本 中 的 一 个 厂 断 ， 因 此 ， 它 目 映 个 
会 产生 任何 输出 。 但 是 ， 这 是 需要 理解 的 一 个 重要 的 代码 段 ， 这 也 是 为 
什么 单独 列 出 它 来 以 便 做 出 一 些 说 明 。 


在 第 2 行 的 f 语 句 中 ， 我 们 测试 表单 是 否 提 供 了 month 和 year。 如 果 
month 和 year 还 没有 定义 ， 代 人 上 段 中 和 后 所 使 用 的 mktime() 函 数 无 法 从 末 
定义 的 month 和 year 参 数 生 成 一 个 日 期 。 


如 果 这 些 值 已 经 有 了 ， 我 们 在 第 3 行使 用 getdate() 来 根据 当前 时 间 创 
建 一 个 关联 数组 。 随 后 ， 我 们 使 用 数组 的 mon 和 year 元 素来 设置 $month 
和 $year 的 值 〈 第 4 行 和 第 5 行 ) 。 如 果 已 经 从 表单 设置 了 这 些 变 量 ， 我 
们 把 数据 放 入 到 $month 和 和 $year 变量 中 ， 人 免得 去 动 最 人 切 的 超 全 局 变量 
$_POST 中 的 值 。 


既然 我 们 可 以 确保 在 $month 和 $year 中 有 了 有 效 的 日 期 ， 束 可 以 使 
用 mktimeO 来 为 该 月 的 第 一 天 创建 一 个 时 间 礁 (第 10 行 )。 随 后 还 需要 
有 天 这 一 时 间 稚 的 信息 ， 因 此 ， 在 第 11 行 ， 我 们 创建 了 一 个 名 为 
$firstDayArray 的 变量 ， 它 将 存储 getdate0 根 据 这 一 时 间 惟 所 返回 的 一 个 
天 联 数 组 。 


24.1.2 构建 HTML 表 单 


现在 需要 创建 一 个 界面 ， 以 便 可 以 让 用 户 看 到 一 个 月 份 或 一 年 的 日 
期 。 为 此 ， 我 们 使 用 了 SELECT 元 素 。 尽 管 可 以 用 HTML 直接 编码 ， 但 
征 我 们 必须 确保 下 拉 列 表 默 认 的 是 当前 选择 的 月 份 ， 因 此 ， 将 动态 地 创 
建 这 一 下 拉 列 表 ， 通 过 向 相应 的 OPTION 元 素 添 加 一 个 SELECT 属性 。 
表单 在 程序 清单 24.2 中 产生 。 


程序 清单 24.2 ”为 日 历 脚 本 构建 HTML 表单 


<?php 
if ({!isset($ POST['month']}} || (tisset($_POST['year'])}}) { 


$nowArray = getdate{}; 
$month = $nowArray['mon' ]; 
$year = $nowArray['year']; 
} else { 
$month = $ POST[ month’ ]; 
$year = $ POST['year']; 
} 


: $start = mktime (12, ©, 0, $month, 1, $year); 
: $firstDayArray = getdate($start); 


> 

<{DOCTYPE html> 

<html> 

<head> 

<title><?php echo "Calendar:".$firstDayArray['month']." 
". $firstDayArray[ ‘year']; ?></title> 

<body> 

<hi>Select a Month/Year Combination</h1> 

<form method="post" action="<?7php echo $ SERVER['PHP SELF']; ?>"> 

<select name="month'"> 

<?php 


: $months = Array("January", "February", "March", "April", "May", 


"June", "July", "August", "September", "October", "November", "December'") ; 
for ($x=1; $x <= count{$months); $x++) { 
echo"<option value=\"$x\""; 
if ($x == $month) { 
echo " selected"; 


i 
echo ">" .$months[$x-1]."</option>"; 
} 
?> 
</select> 
<select name="year"> 
<?php 


for ($x=1990; $x<=2020; $x++) { 
echo "<option"; 
if ($x == $year) 1 
echo " selected"; 
} 
echo ">$x</option>"; 


} 


> 
: go hh Se 
< 


utton type="submit" name="submit" value="submit">Go!</button> 


: </form> 
: </body> 
: </html> 
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行 ， 让 我 们 开始 同 页 面 写 HTML。 注 意 ， 在 第 15 行 和 第 16 行 ， 我 们 使 用 
$firstDayArray 来 把 月 份 和 年 份 添加 到 TITLE 元 又 。 


第 20 行 是 表单 的 开始 。 要 为 月 份 下 拉 列 表 创 建 SELECT 元 系 ， 我 们 
在 第 22 行 义 回 到 了 了 PHP 模式 ， 来 编写 单个 的 OPTION 标 记 。 首 先 ， 我 们 
在 第 23 行 到 第 24 行 创建 了 一 个 名 为 $months 的 数组 ， 其 中 包含 了 12 个 月 
份 的 名 字 ， 以 便 用 来 显示 。 然 后 ， 我 们 遇 历 了 这 个 数组 ， 为 每 个 名 字 创 
建 了 一 个 OPTION 标 记 ( 第 25 行 到 第 31 行 )。 


如 果 不 是 我 们 要 在 第 27 行 根据 $month 变 量 测试 $x (for 语 句 中 的 计 
数 变 量 ) 的 话 ， 这 真是 编写 一 个 简单 SELECT 的 过 于 复 林 的 方法 。 如 果 
$x 和 $month 相等 ， 我 们 把 字符 串 SELECTED 添 加 到 OPTION 标 记 ， 以 
确保 页 面 载 入 的 时 候 正 确 的 月 份 裤 上 自动 选取 。 在 第 36 行 到 第 42 行 ， 我 们 
使 用 一 种 类 似 的 技术 来 把 年 份 写 入 到 下 拉 列 表 中 。 最 后 ， 回 到 HTML 模 
式 ， 我 们 在 第 45 行 创建 了 一 个 提交 按钮 。 


现在 ， 我 们 有 了 一 个 表单 可 以 同 它 目 己 发 送 month 和 year 参 数 ， 并 
且 要 么 默认 为 当前 的 月 份 和 年 份 ， 要 么 是 之 前 选取 的 年 份 和 月 份 。 如 采 
我 们 把 这 个 程序 清单 保存 为 dateselector.php， 将 其 放置 到 Web 服 务 器 文 
档 根 目录 下 ， 然 后 使 用 Web 浏 览 右 访问 它 ， 将 会 看 到 如 图 24-1 所 示 的 结 
果 【〈 你 的 月 份 和 年 份 可 能 不 同 〉。 


eee 
(©) Calendar: March 2012 


Z C © http://localhost/24/dateselector.php O r aS 


Select a Month/Year Combination 
March [e] |2012 [æ] 


图 24-1 日历 表单 


24.1.3 ”创建 日 历 表格 

现在 需要 创建 一 个 表格 ， 并 且 使 用 选 定 的 月 份 的 日 期 来 填充 它 。 我 
们 在 程序 清单 24.3 中 做 到 这 一 点 ， 这 是 一 个 完整 的 日 历 显 示 脚 本 。 

尽管 第 2 行 是 新 内 容 ， 但 是 第 3 行 到 第 64 行 完全 来 和 目 于 程序 清单 
24.2。 第 2 行 只 是 定义 了 一 个 第 量 ， 在 这 个 例子 中 整 是 值 为 86400 的 
ADAY (例如 ,“a day”)。 这 个 值 表示 一 天 之 中 的 秒 数 ， 我 们 将 在 这 个 脚本 
的 后 耐用 到 它 。 

程序 清单 24.3 ”完整 的 日 历 显示 脚本 


On Oooh — 


<?php 
define("ADAY", (60*60*24)}; 
if ({!isset($ POST['month'])) || (!isset({$ POST['year']})) { 
$nowArray = getdate(); 
$month = $nowArray['mon' J; 
$year = $nowArray['year']; 
} else { 
$month = $ POST['month' J; 
$year = $ POST['year']; 
} 


: $start = mktime (12, ®, 8, $month, 1, $year); 
: SfirstDayArray = getdate($start); 


?> 
<!DOCTYPE html> 
<html> 
<head> 
<title><?php echo "Calendar: ".$firstDayArray['month']." 
". $firstDayArray['year'']; ?></title> 
<style type="text/css"> 
table { 
border: ipx solid black; 
border-collapse: collapse; 


} 
th { 
border: ipx solid black; 
padding: 6px; 
font-weight: bold; 
background: #ccc; 
} 
td { 
border: 1px solid black; 
padding: 6px; 
vertical-align: top; 
width: 100px; 
} 
</style> 
<body> 


<hi>Select a Month/Year Combination</h1> 
<form method="post" action="<?php echo $ SERVER['PHP_SELF']; ?>"> 
<select name="month"> 
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<?php 
¢months = Array("January", "February", 
"June", “July", "August", "September", 


"March", "April", 


"October", 


for {$x=1; $x <= count($months}; $x++) { 


echo"<option value=\"$x\""; 
if ($x == $month) { 
echo " selected"; 


} 
echo ">". $months[$x-1].°</option>"; 
} 
?> 
</select> 
<select name="year"> 
<?php 


for ($x=1980@; $x<=201@; $x++) { 
acho “Sant Tens 
if ($x == $year) { 
echo " selected"; 
} 
echo ">$x</option>"; 


} 


?> 
</select> 


"November", 


i May ul : 


"December"}; 


<button type="submit" name="submit" value="submit">Go!</button> 


</form> 
<br/> 
<?php 


$days = Array( "Sun", "Mion", "Tue", "Wed", THUS, 


echo "<table><tr>\n"; 
foreach ($days as $day) { 

echo "<td>" .$day.</td>\n"; 
} 


for ($count=@; $count < (6*7): $countt+) { 


$dayArray = getdate($start}); 
if (($count % 7) == 6) { 


if ($dayArray['mon'] != $month) { 


break; 
} else { 
echo "</tr><tr>\n"; 
} 
} 
if ($count < $firstDayArray[ ‘wday' ] 
echo "<td>&nbsp;</td>\n"; 
} else { 


|| $dayArray['mon'] 


echo "<td>". $dayArray['mday']."</td>\n"; 


$start += ADAY; 


} 
} 
echo "</tr></table>"; 
?> 
</body> 


</html> 


“Epa” ; 


"Sat" ) : 


l= $month) { 


我 们 在 第 66 行 看 到 新 代码 。 由 于 这 个 表格 将 按照 星期 几 来 索引 ， 我 
们 在 第 70 行 到 第 72 行 过 历 星 期 几 名 称 的 一 个 数组 ， 在 第 71 行 ， 将 每 个 名 
称 显 示 在 其 自己 的 表格 单元 格 中 。 这 个 脚本 的 所 有 神奇 之 处 在 于 最 后 ， 
即 从 第 73 行 开始 的 语句 。 


在 第 73 行 ， 我 们 初始 化 一 个 名 为 $count 的 变量 ， 并 且 确 保 循 环 和 在 第 
42 座 欠 代 之 后 结束 。 这 融 确 体 了 将 有 足够 的 单元 格 来 填充 日 期 信息 ， 考 
虑 a 到 包含 4 个 星期 的 一 个 月 可 能 实际 在 月 初 和 月 末 还 有 几 天 ， 因 此 ， 需 
要 6 个 7 天 的 周 ( 行 ) 。 


在 这 个 for 循 环 中 ， 我 们 使 用 getdate() 把 $start 变 量 转 换 为 一 个 日 期 数 
组 ， 把 结果 赋 给 $dayArray (第 73 行 )。 尽 管 在 循环 的 最 初 执行 中 ， 
$start 是 该 月 的 第 一 天 ， 对 于 每 次 欠 代 ， 我 们 将 使 用 ADAY 〈24 小 时 ) 的 
值 来 递增 这 个 时 间 惟 《参见 第 85 行 ) 。 


在 第 75 行 ， 我 们 使 用 醒 除 操作 符 根 据 数 字 7 来 测试 $count 变 量 。 只 
有 妆 $count 是 0 或 者 是 7 的 倍数 的 时 候 ， 属 于 这 个 站 语句 的 代 人 码 块 才 会 运 
行 。 通 过 这 种 方式 ， 我 们 惑 知道 了 应 该 结束 循环 或 者 开始 一 个 新 的 行 ， 
这 里 ， 行 表示 周 。 


当 我 们 已 经 确定 处 于 第 一 次 迭代 中 或 者 到 达 了 一 行 的 末尾 ， 可 以 继 
续 在 第 76 行 执行 另 一 个 测试 。 如 果 $dayArray 的 mon (月 份 编 写 ) RA 
等 于 $month 变 量 ， 我 们 就 结束 了 。 别 忘 了 ，$dayArray 包 含 了 有 关 
$start 时 间 惟 的 信息 ， 这 就 是 我 们 所 显示 的 月 份 的 当前 位 置 。 当 $start 超 
出 了 当前 月 份 的 范围 ，$dayArray['mon'"] 将 会 保存 和 用 户 输 入 所 提供 的 
$month 不 同 的 一 个 数字 。 我 们 的 模 除 测试 显示 ， 到 达 了 一 行 的 末尾 ， 并 
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示 的 月 份 中 ， 我 们 在 第 79 行 结束 一 行 并 开始 新 的 一 行 。 


在 第 82 行 下 一 条 让 语 句 中 ， 我 们 确定 了 是 售 把 日 期 信息 与 入 到 一 个 
单元 格 。 并 不 是 每 个 月 份 都 从 周 日 开始 ， 因 些 ， 行 很 可 能 包 侣 一 个 或 两 
个 衬 单 元 格 。 关 似 的 ， 很 少 有 月 份 刚 好 在 一 行 末 尾 结束 ， 因 此 ， 在 结 
表格 前 可 能 有 几 个 空 单元 格 。 


我 们 已 经 把 和 一 个 月 份 的 第 一 天 相关 的 信息 存储 到 $firstDayArray 
中 ， 特 别 是 可 以 访问 $firstDayArray["wday] 中 表示 星期 几 的 数字 了 。 如 
果 $count 的 值 比 这 个 数字 小 ， 我 们 知道 还 没有 到 达 要 写 入 的 正确 的 单元 
格 。 以 此 类 推 ， 如 果 $month 变 量 的 值 不 再 等 于 $dayArray['mon']， 我 们 知 
道 到 达 了 月 末 ， 但 还 没有 a 到达 行 末 ， 正 如 我 们 在 前 面 的 模 际 测试 中 所 判 
定 的 那样 。 在 这 两 种 情况 下 ， 我 们 都 在 第 83 行 网 浏览 器 中 与 入 一 个 空 单 
元 格 。 


在 第 67 行 的 最 后 一 个 else 子 名 中 ， 我 们 可 以 做 一 些 有 趣 的 事情 。 我 
们 已 经 确定 在 想 归 列 出 的 月 份 之 中 ， 并 且 当 前 日 期 列 和 
$firstDayArray["wday] 中 存储 的 星期 几 的 数字 十 相等 的 。 现 在 ， 必 须 使 
用 前 面 在 循环 中 建立 的 $dayArray 天 联 数组 来 把 月 份 中 的 日 期 和 一 些 衬 
白 写 入 到 一 个 单元 格 中 。 


最 后 ， 在 第 69 行 需要 递增 $start 变 量 ， 有 其 中 包含 了 时 间 戳 。 我 们 只 
征 将 其 增加 一 天 的 秒 数 〈 在 第 2 行 定 义 了 这 个 数字 ) ， 并 且 ， 我 们 准备 
好 再 次 测试 $start 中 的 新 值 来 开始 循环 。 如 采 把 这 个 程序 清单 体 存 为 
showcalendar.php， 将 其 放置 到 Web 服 务 需 的 文档 根 目录 下 ， 并 且 通 过 
Web 浏 览 右 来 访问 它 ， 将 会 看 到 如 岁 24-2 所 示 的 结 末 《你 的 月 份 和 年 份 


可 能 有 所 不 同 ) 。 
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Q Calendar: March 2012 
ie) Fe ad 


€ > CO http://localhost/24/showcalendar.php 


Select a Month/Year Combination 


March | 2012 |=| Gol | 








图 24-2 日历 表单 和 脚本 


24.1.4 [WA ase 

显示 日 历 是 不 错 的 事情 ， 但 是 ， 只 需要 略 多 几 行 的 代码 ， 束 可 以 让 
日 历 更 具有 交互 性 ， 也 就 是 说 ， 我 们 可 以 在 给 定 的 一 天 添加 和 浏览 
件 。 要 开始 做 到 这 一 点 ， 让 我 们 创建 一 个 简单 的 数据 库 表 ， 其 中 存储 了 
事件 信息 。 为 了 人 徐 音 起见， 这 些 事件 只 在 单独 的 一 天 发 生 ， 并 且 将 只 显 
示 其 开始 日 期 和 时 间 。 尽 管 我 们 可 以 让 事件 记录 尽 可 能 地 复杂 ， 这 里 的 
这 个 例子 只 是 为 了 展示 所 涉及 的 基本 过 程 。 

calendar_events 表 将 包 侣 用 于 开始 日 期 和 时 间 、 事 件 标 题 以 及 事件 
的 简短 说 明 的 字段 。 


CREATE TABLE calendar_events { 
id INT NOT NULL PRIMARY KEY AUTO_INCREMENT, 
event title VARCHAR (25), 
event shortdesc VARCHAR (255), 
event start DATETIME 


我 们 可 以 使 用 程序 清单 24.3 中 的 代码 作为 基础 。 在 新 的 脚本 中 ， 将 
浓 加 一 个 到 弹出 窗口 的 链接 作为 日 历 显示 的 一 部 分 。 每 个 日 期 都 是 一 个 
链接 ， 弹 出 窗口 将 调用 另外 一 个 脚本 ， 访 脚本 显示 一 个 事件 的 完整 文本 
并 且 能 够 添加 事件 。 为 了 开始 做 到 这 一 点 ， 先 在 最 初 的 脚本 的 第 36 行 的 
开始 <head> 标 记 的 后 面 ， 添 加 如 下 的 JavaScript 代 人 码 。 


<script type="text/javascript'"> 
function eventWindow(url) { 

event_popupWin = window.open(url, ‘event', ‘resizable=yes, scrollbars=yes, 

toolbar=no,width=4@@,height=400' ); 

event _popupWin.opener = self; 
} 
</script> 

这 个 JavaScript 函 数 定 义 了 一 个 400x400 的 窗口 ， 它 将 调用 我 们 所 提 

供 的 一 个 URL。 在 最 饭 脚 本 的 第 85 行 使 用 这 个 JavaScript 函 数 ， slain 
把 这 个 链接 中 显示 的 日 期 包装 到 基于 JavaScript 的 弹出 窗口 中 ， 这 个 窗口 


将 调用 一 个 名 为 event.php 的 脚本 。 新 的 代码 行 如 下 所 示 。 


echo "<td><a href=\"javascript:eventWindow({('event.php?m=". $month. 
"&Samp;d=".$dayArray['mday']."&amp;y=$S$year'};\">".$dayArray['mday']."</a> 
<or/><br/>".Gevent title."</td>\n"; 
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送 3 个 变量 ， 这 就 是 针对 月 份 的 $_GET['m']， 针 对 日 期 的 $_GET['d']， 和 
针对 年 份 的 $_GETI['y']。 


在 我 们 人 处理 event.php 脚 本 之 前 ， 对 于 这 个 特定 的 脚本 只 保留 了 一 点 


更 改 : ORS PSE FETE, RII ME SI SN ak X 
(SA ir Be Per Ee AINA SE, “ELE ee I) AAS 23 8477 Helse 
语句 的 开始 。 出 现 了 一 个 全 新 的 else 语 句 ， 我 们 可 以 看 到 查询 在 这 里 执 
行 了 ， 并 且 ， 如 来 找到 结果 ， 文 本 将 在 那 一 天 的 表格 单元 格 中 喧 示 出 

来 。 


} else { 
$mysqli = mysqli_connect("localhost", "joeuser", "somepass", "testDB"); 
$chkEvent_sql = "SELECT event_title FROM calendar events WHERE 
month(event_start} = °".$month."' AND 
dayofmonth{event start} = '".$dayArray['mday']."' 
AND year(event_ start) = '".$year."' ORDER BY event start"; 


$chkEvent_res = mysqli _query($mysqli, $chkEvent_sql) 
or die(mysqli_error($mysqli)}; 


if (mysqli num rows($chkEvent res) > 6) { 
while ($ev = mysqli fetch array($chkEvent res}) 1 
$event title = stripslashes($ev['event_title']); 
} 
} else { 


$event title = ""; 
} 


echo "<td><a href=\"javascript:eventWindow( event. php?m=".$month. 
Ii &amp ; d=" .$dayArray [ 'mday' ] "&amp ;y=$year ' ) : \ ws à 
$dayArray['mday' ]."</a><br/><br/>".$event_title."</td>\n"; 
unset($event title); 


$start += ADAY; 


在 程序 清单 24.4 中 ， 我 们 可 以 看 到 完整 的 新 的 脚本 ， 名 为 


showcalendar_withevent.php. 
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<?php 
define("ADAY", (60*60*24))}; 
if {({!isset($ POST['month'])) || (!isset($ POST['year']))) { 
$nowArray = getdate(); 
$month = $nowArray['mon' ]; 
$year = $nowArray['year']; 
} else { 
$month = $ POST['month']; 
$year = $ POST['year']; 
} 
$start = mktime (12, @, ©, $month, 1, $year); 
$firstDayArray = getdate($start) ; 
?> 


: <IDOCTYPE html> 


<html> 
<head> 
<title><?php echo "Calendar: ".$firstDayArray['month']." ". 
$firstDayArray[ 'year'] ?></title> 
<head> 
<style type="text/css"> 
table { 
border: 1px solid black; 
border-collapse: collapse; 
} 
tnd 
border: 1px solid black; 
padding: 6px; 
font-weight: bold; 
background: #ccc; 


} 
td a 


32: 
Ke Fe 
34: 
3a; 
eos 
as 
38: 
39: 
4@: 
41: 
42: 
43: 
44: 
45: 
46: 
47: 
48: 
49; 
5@: 
Li 
52: 
5g., 
54: 
55: 
56: 
BFS 
58: 
59: 
60: 
61: 
62: 
63: 
64: 
659: 
66: 
67: 
68: 
69: 
70: 
7 Ls 
fe: 
73: 
74: 
Poa 
fos 
Tia 
78: 
79: 
88: 
B81 
82: 
83: 
84: 
85: 
86: 
87: 
88: 
89: 


border: ipx solid black; 
padding: 6px; 
vertical-align: top; 
width: 10@px; 
} 
</style> 
<script type="text/javascript"> 
Function eventWindow({url) { 
event_popupWin = window.open{url, ‘event’, 
‘'resizable=yes,scrollbars=yes, toolbar=no,width=40@, height=468' ); 
event popupWin.opener = self; 
f 
</script> 
<body> 
<hi>Select a Month/Year Combination</h1> 
<form method="post" action="<?php echo $ SERVER['PHP_SELF']; ?>"> 
<select name="month"> 


<?php 
$months = Array ¢ "January" j il February ll P "March" ; "April" P "May" 3 "June" i 
"July", "August", "September", "October", "November", "December" }; 


for ($x=1; $x <= count(S$months}; $xt++) { 
echo"<option value=\"$x\""; 
if ($x == $month) { 
echo " selected"; 
} 
echo ">".$months[$x-1]."</option>"; 
i 
a 
</select> 
<select name="year"> 
<?php 
for ($x=1990; $x<=2020; $x++) { 
echo "<option"; 
if ($x == $year) { 
echo " selected"; 
} 
echo ">$x</option>"; 
} 
?> 
</select> 
<button type="submit" name="submit" value="Submit">Go!</button> 
</form> 
<br/> 
<?php 
$days = Array(*Sun", "Mon", "Tue", "Wed", “Thu", "Fri”, "Sat"); 
echo "<table><tr=>\n"; 
foreach ($days as $day) { 
echo "<th>$day</th>\n"; 
} 


for ($count=@; $count < (6*7); $count++} { 
$dayArray = getdate($start}; 


if (($count % 7) == @) { 
if ($dayArray['mon'] != $month} { 
break; 
+ else { 


echo "</tr><tr>\n" ; 


} 


90: 
91: 
92: 
93: 
94: 


95; 
96: 
97: 
98: 


99: 

190: 
181: 
102: 
193: 
104; 
105: 
196: 
107: 
198: 
199: 
119: 
TII: 
112: 
113 
114: 
115: 
116: 
TT 
118: 
119: 
129: 
121: 
122: 
lee 
124; 


} 
if ($count < $firstDayArray['wday'] || $dayArray['mon'] != $month) 1 
echo "<td>&nbsp;</td>\n°' ; 


} else { 
$mysqli = mysqli_connect("localhost", “joeuser", "“somepass", 
"testDB" }; 
$chkEvent_sql = "SELECT event_title FROM calendar_events WHERE 
month(event start) = '".$month."' AND 
dayofmonth{event start) = ‘'".$dayArray['mday']."' 
AND year{event_start) = '".$year."' ORDER BY 


event start"; 
$chkEvent res = mysqli query($mysqli, $chkEvent sql) 
or die(mysqli_error($mysqli) }; 


if (mysqli_num_rows({$chkEvent_res} > @} { 
while ($ev = mysqli fetch array($chkEvent_res}} { 
$event title .= stripslashes($ev['event_title']."<br/>'"; 
} 
} else 1 
$event title = ""; 


} 


echo "<td><a href=\"javascript:eventWindow('event.php?m=". $month. 
"gamp;d=".$dayArray['mday'] ."&amp;y=$year');\">". 
$dayArray['mday']."</a><br/><br/>".$event_title."</td>\n"; 


unset{$event_title}; 
$start += ADAY; 
} 
} 


echo "</tr></table>"; 


/iclose connection to MySQL 
mysqli close($mysqli}; 

?> 

</body> 

</html> 


在 图 24-3 中 ， 我 们 可 以 看 到 新 的 日 历 ， 包 括 在 填写 了 一 个 事件 的 日 


期 上 显示 事件 标题 ， 为 了 便于 说 明 ， 在 这 里 ， 我 们 在 calendar_events 表 
中 增加 了 一 个 事件 。 


(©) Calendar: March 2012 
hace F a8 


r oisthoront eke 
rWIiineventpnp 


过 C | © http://localhost/24/showcalenda 


Select a Month/Year Combination 


5s R 
18 19 20 

Go to the Zoo 

üj 





图 24-3 ”显示 带 有 事件 的 日 历 


剩 下 的 就 只 是 添加 一 体 化 的 event.php 脚 本 ， 这 个 脚本 用 于 弹出 窗口 
的 显示 并 且 也 添加 一 个 事件 到 日 历 〈 在 一 个 特定 的 日 期 ) 。 程 序 清单 
24.5 包 含 了 所 有 上 所 需 的 代码 ， 有 趣 的 部 分 从 第 8 行 开 始 ， 那 里 连接 到 
MySQL 数 据 库 。 第 11 行 得 看 事件 条 目 表 单 是 含 已 经 提交 ， 如 采 已 经 提 
区 ， 在 第 14 行 到 第 24 行 创建 了 对 数据 库 安 全 的 什 ， 创 建 并 执行 一 条 
INSERT 语 句 ， 同 calendar_events 表 添加 事件 ， 然 后 再 继续 执行 (第 29 行 
到 第 34 行 ) 。 

程序 清单 24.5 ”通过 弹出 窗口 显示 事件 /添加 事件 


0 s ee E eb c 


iO 


<IDOCTYPE html> 


<html> 
<head> 
<title>Show/Add Events</title> 
<body> 
<hi=show/Add Events</h1> 
<?php 
$mysqli = mysqli connect("localhost", ‘joeuser", ‘somepass", “testDB"}; 
ffadd any new event 
if ($ POST) { 
/fcreate database-safe strings 
$safe m = mysqli real escape string($mysqli, $ POST['m']); 
Fsafe d = mysqli real escape string($mysqli, $ POST[ d 1]}); 
$safe y = mysqli real escape string($mysqli, $ POST[ y']); 
$safe event title = mysqli real escape string($mysqli, 
$ POST[ event title']}; 
$safe_event_shortdesc = mysqli_real_escape string($mysqli, 
$ POST[ 'event shortdesc' ]); 
$Safe event _time hh = mysqli real _ escape string($mysqli, 
$ POST['event time hh']}; 
中 Safe event time mm = mysqli real escape string({$mysoli, 
$ POST[ event 七 Ime mm ]); 
$event date = $safe y."-".$safe m."-".$safe d." 
".$safe event time hh.":".$safe event time mm.":00"; 
finsEvent sql = "INSERT INTO calendar events (event title, 
event_shortdesc, event start) VALUES 
('".Gsafe event title."', '".$safe event shortdesc."', 


"$event date. '')"; 


$insEvent res = mysqli query($mysqli, $insEvent sql) 


34: or die(mysqli error($mysqli)}; 


Soe 

36: } else f 

or: 

38: { {create database-safe strings 

39: $safe m = mysqli _ real escape string{$mysgli, $ GET[ m ]); 
4@: $safe d = mysqli real escape string{$mysqli, $ GET['d']); 
41: $safe y = mysqli real escape string{$Smysgli, S$ GET[ y ]) 
42: } 

43: 

44: //show events for this day 

45: S$getEvent sgl = "SELECT event title, event shortdesc, 

46: date format(event start, ‘%l:%1 %p') as fmt date 
47: FROM calendar events WHERE month(event start) = 
48: '", $safe_m."' AND dayofmonth{event_start} = 

49 : '" $safe d."' AND year(event start) = 

5@: '" Ssafe_y."' ORDER BY event _start"; 

51: SgetEvent_ res = mysqli _query($mysqli, $SgetEvent sql) 

52: or die(mysqli_error($mysqli)); 

59s 

54: if (mysqli num rows($getEvent res) > Ø} { 

55: $event txt = "<ul>"; 

56: while ($ev = @mysqli fetch array($getEvent res}) { 

57: $event title = stripslashes($ev[‘event_title']); 

58: $event shortdesc = stripslashes{$ev['event shortdesc']}; 
59: Sfmt date = $ev['fmt date']; 

BO: $event txt .= "<1li><strong>".$fmt_date."</strong>: 

61: ", $event_title. "<br/>" .$event_shortdesc,."</li>'; 
Ge: } 

63: Sevent txt .= "</ul>"; 

64: mysqli free result(SgetEvent res}; 

65: } else 1 

66: $event txt = ""; 

67: } 


68: // close connection to MySQL 
69: mysqli close($mysq1i}; 


7@; 

71: aif ($event_txt I= ""} { 

72: echo "“<p><strong>Today's Events:</strong></p> 
fo: f$event txt 

74: <nar fe"; 

foe: ip 

ro: 


77: fi show form for adding an event 

78: echo <<<END OF TEXT 

79: <form method="post" action="$ SERVER[PHP_SELF] > 

80: <p><strong>Would you like to add an event?</strong><br/> 
81: Complete the form below and press the submit button to 
82: add the event and refresh this window.</p> 

33: 


84: <p><label for="event_title">Event Title:</label><br/> 

85: <input type="text’ id="event title" name="event title" 

86: Size="25" maxlength="25" /></p> 

87: 

88: <p><label for= event shortdesc">Event Description: </label><br/> 
89: <input type="text" id="event_shortdesc" name="event_shortdesc" 
90: size="25" maxlength="255" /></p> 

91: <fleldset> 

92: <legend>Event Time (hh:mm}:</legend> 

93: <select name="event_time hh > 

94: END OF TEXT; 


95: 

96: for ($x=1; $x <= 24; $x++) { 

97: echa "<option value=\"$x\">$x</aotion>"; 
98: } 

99: 


180: echo <<<END OF TEXT 

101: </select> : 

102: <select name="event_time mm > 

193: <option value="90">60</option> 

1404: <option value="15°>15</option> 

1@5: <option value="3@°>38</option> 

186: <option value="45">45</option= 

107: </select> 

198: </fieldset> 

1@9: <input type="hidden" name="m" value="$safe_m"> 
110: <input type="hidden" name="d" value="Ssafe d"> 
111: <input type="hidden" name="y" Value= $sate ¥ > 
112: 

113: <button type="submit" name="submit" value="sSubmit">Add Event</button> 
114: </form> 

115: END OF TEXT; 

116: ?> 

117: </body> 

118: </html> 


FAST PAA S21T PUT SAFE HIREA S ZS EH A SET DY. 
的 所 有 记录 。 用 来 显示 条 目的 文本 块 在 第 54 行 到 第 67 行 创建 。 然 而 ， 用 
尸 也 需要 看 到 漆 加 一 个 事件 的 表单 ， 并 且 这 个 表 早 在 第 79 行 到 第 114 行 
创建 ， 实 际 上 也 束 古 这 个 脚本 的 末尾 。 


在 图 24-4 中 ， 我 们 看 到 了 当 从 日 历 打 开 一 个 链接 的 时 候 ， 一 个 弹出 
和 贸 口 是 什么 样 的 ， 并 且 喧 示 了 一 个 条 目 。 在 这 个 例子 中 ， 我 们 想 要 在 这 
一 天 添加 为 外 一 个 事件 ， 因 此 ， 这 个 表 蛙 已 经 完成 并 准备 好 添加 故 一 个 
事件 。 


@ Show/Add Events - Google Chrome 


©) localhost/24/event.php?m=34d=1h &y=201; 


Show/Add Events 
Today's Events: 
+ 1:00 PAL Go to the Zoo 


Time to see the Red Pandas! 


Would you like to add an event? 
Complete the form below and press the submit button to add the event and 
refresh this window. 


Event Title: 
Event Descnpton: 


m Event Time (hhmm): 
1 [=] -/00[+| 


Add Event | 





图 24-4 eAN SIR INAS, ERRE. 


在 图 24-5 中 ， 在 这 个 特定 的 日 期 讨 加 了 第 二 个 事件 。 


| @ Show/Add Events - Google Chrome 


© localhost/24/even t.php 


Show/Add Events 


Today's Events: 


+ 1:00 PM. Go to the Zoo 
Time to see the Red Pandas! 

© 4:00 PM. Leave the Zoo 
Time to go home. x( 


Would vou like to add an event? 
Complete the form below and press the submit button to add the event and 
refresh this window. 


Event Title: 


Event Description 


Event Time (hh:mm): 


Add Event 





图 24-5 ”已 经 添加 了 第 二 个 事件 


显然 ， 这 是 一 个 简单 的 例子 ， 但 是 它 展示 了 构建 一 个 日 历 类 型 的 系 
统 实际 上 很 容易 ， 只 需要 一 个 简短 的 脚本 。 


24.2 ”创建 一 个 日 历 库 


由 于 日 期 在 Web 界 面 上 是 随处 可 见 的 ， 并 且 由 于 使 用 日 期 往往 相对 
比较 复杂 ， 让 我 们 看 看 如 何 创建 一 个 关 库 来 目 动 化 东 些 日 期 巡 示 的 工 
作 。 在 这 个 过 程 中 ， 我 们 将 会 回顾 条 些 已 经 介绍 过 的 搁 术 ， 尤 其 是 在 第 
9 草 中 介绍 过 的 内 容 。 


这 个 实例 中 创建 的 徐 单 的 date_puldown 库 将 由 3 个 分 开 的 Select 元 对 
组 成 ， 一 个 用 于 月 份 中 的 日 期 ， 一 个 用 于 月 份 ， 一 个 用 于 年 份 。 


当 用 户 提 交 一 个 页 面 ， 这 个 脚本 将 验证 雪 单 输入 。 如 琳 输 入 没有 问 
题 ， 我 们 将 使 用 仍然 存在 的 用 户 和 输入 来 刷新 这 个 页 面 。 便 用 文本 框 ， 这 
RAD TEM, (AGA RIA SAL. NM PEE is 
FA ES) A TE) tS Mimi Ae SAY Pd el. HH DE cia A BSC AS SE A 
系 的 值 属 性 中 。 日 期 将 需要 划分 为 月 份 、 日 期 和 年 份 什 ， 以 及 选中 的 正 
确 的 选择 元 系 。 


date_pulldown 类 的 目的 束 是 使 得 日 期 下 拉 亲 单 稳 定 〈 这 样 ， 它 们 可 
以 在 页 面 和 页 面 之 间 保 存 设置 ) 并 且 容 易 设置 。 为 了 创建 我 们 的 类 ， 首 
先 需 要 声明 它 并 且 创 建 一 个 构造 方法 。 


提示 : 


构造 方法 存在 于 类 中 ， 并 且 当 创建 类 的 一 个 新 的 实例 的 时 候 目 动 调用 该 函数 。 


我 们 也 可 以 声明 一 些 类 属性 。 我 们 将 浏览 程序 清单 24.6， 这 走时 未 


类 的 开始 的 第 一 个 代码 段 。 


程序 清单 24.6 ”创建 一 个 日 历 库 
1: class date pulldown { 
2 public $name; 
3 public $timestamp = -1; 
4 public $months = array("Jan", "Feb", "Mar", "Apr", "May", "Jun", 
S: ‘with. “AUG. “Sen Ger “Now, “Dete 
6: public $yearstart = -1; 
Ti public $yearend = -1; 
8 
9 function date pulldown($Sname} { 
四 
1 


$this->name = $name; 
} 


首先 在 第 2 行 声 明了 $name 属 性 ， 这 将 用 来 命名 HTML select tz. 
第 3 行 定 义 的 $timestamp 属 性 ， 将 存储 一 个 UNIX 时 间 戳 。 第 4 行 到 第 5 行 
定义 的 $months 数 组 属性 ， 包 含 了 我 们 在 月 份 下 拉 列 表 中 显示 的 字 稚 
串 。$yearstart 和 $yearend 属 性 (第 6 行 和 第 7 行 ) 初始 都 设置 为 -1， 它 们 
最 终 将 保存 年 份 下拉 列 表 中 显示 的 第 一 年 和 最 后 一 年 的 学 围 。 


构造 方法 是 简单 的 。 它 接受 一 个 字符 串 ， 即 我 们 赋 给 $name 属 性 的 
字符 串 。 既 然 我 们 已 经 有 了 类 的 基础 ， 还 需要 一 组 方法 ， 以 便 客 户 代 码 
通过 它们 能 够 设置 日 期 。 程 序 清单 24.6 从 下 面 继续 。 


(i 


程序 清单 24.6 ”创建 一 个 日 历 库 〈 续 ) 


Tz: function setDate global( ) { 


Ta: if ({!$this->setDate array($GLOBALS[$this->name])) { 
14: return $this->setDate timestamp(time{)); 
ite } 

t6: return true; 

Te } 

LO: 

19: function setDate timestamp(Stime) { 

20: $this->timestamp = $time; 

2: return true; 

22: } 

Ere i 

24: function setDate array{$inputdate}) { 

25: if {is_array{$inputdate) && 

26: isset($inputdate['mon']) && 

27: isset{$inputdate['mday']) && 

28: isset($inoutdate['year']}} { 

29: 

30: $this->timestamp = mktime(11, 59, 59, 
a1 $inputdate['mon'], $inputdate['mday'], $inputdate['year']}; 
ene return true; 

33: 

34: return false; 

S53 } 


在 这 里 所 给 出 的 方法 中 ，setDate_timestampO 是 最 简单 的 〈 第 19 行 
到 第 22 行 ) 。 它 需要 一 个 UNIX 时 间 惟 ， 这 个 时 间 戳 会 赋 给 $timestamp 属 
性 。 但 是 ， 我 们 也 不 要 起 记 了 其 他 的 方法 。 


setDate_array() 方 法 (第 24 行 到 第 35 行 ) 期 竺 一 个 关联 数组 ， 它 至 
少 带 有 3 个 键 : mon、mday 和 year。 这 些 键 所 拥有 的 数据 值 将 会 和 
getdate() 有 所 返回 的 数组 中 的 值 上 其 有 相同 的 格式 。 这 意味 看 setDate_array() 
将 会 接受 一 个 手动 构建 的 数组 ， 示 例如 下 。 


array{ Mday => 25, mon =>3， Year => 2012); 


或 者 是 调用 getdate() 的 结果 ， 如 下 所 示 。 


getdate( 12089052013}; 


这 不 是 偶然 ， 我 们 稍 后 要 构建 的 下 拉 列 表 将 会 产生 一 个 包含 了 


mon、mday 和 year 键 的 数组 。 这 个 方法 使 用 mktime() 函 数 来 构建 一 个 时 
间 鹤 ， 随 后 ， 这 个 时 间 稚 将 赋 给 $timestamp 变 量 。 


setDate_global(0) 方 法 (第 12 行 到 第 17 行 默认 被 调用 。 它 壬 试 找到 
和 对 象 的 gname 属 性 具有 相同 名 字 的 一 个 全 局 变量 ， 这 个 变量 传递 给 了 
setDate_array()。 如 果 这 个 方法 找到 一 个 具有 正确 构造 的 全 局 变量 ， 它 
使 用 这 个 变量 来 创建 gtimestamp 变 量 。 人 否则 ， 使 用 当前 日 期 。 


日 期 和 月 份 的 范围 症 固定 的 ， 但 是 ， 年 份 则 不 同 了 。 随 看 程序 清单 
24.6 的 继续 ， 我 们 创建 了 一 些 方法 ， 人 允许 客户 代码 设置 目 己 的 年 份 范 
围 ， 尽 过 我们 也 提供 了 默认 的 行为 。 


程序 清单 24.6 ”创建 一 个 日 历 库 ( 续 ) 


36: Function setYearStart($year) { 

af Sthis-=yearstart = $year; 

38: } 

38: 

4@: function setYearEnd{$year) { 

A $this->yearend = $year; 

42; } 

Ag: 

44: function getYearStart() { 

45: if ($this->yearstart < @) { 

46: $nowarray = getdate(time(}}); 
Af: $this->yearstart = $nowarray['year']-§; 
48: } 

49: 

50: return $this->yearstart; 

5J; \ 

Bet 

a po function getYearEnd{) { 

54: if ($this->yearend < @) { 

55. Snowarray = getdate(time(}); 
56: Fthis->yearend = $nowarray['year']+5; 
57 } 

58: return $this->yearend; 


setYearStart() 和 setYearEnd() 方 法 都 相当 简单 (第 36 行 到 第 43 行 )〉， 
为 一 个 年 份 直 接地 赋 给 了 相应 的 属性 。getYearStart(0) 方 法 测试 
$yearstart 属 性 是 任 已 经 设置 了 ， 如 末 还 没有 人 设置，getYearStart() 把 当前 
年 份 之 前 5 年 的 值 赋 给 $yearstart。getYearEnd() 方 法 执行 类 似 的 操作 。 


=j 
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程序 清单 24.6 ”创建 一 个 日 历 库 ( 续 ) 


60: function output() { 


61: if ($this->timestamp < @) { 

62: $this->setDate global(); 

63: } 

64: $datearray = getdate($this->timestamp) ; 

65: Gout = $this->day_select($S$this->name, G$datearray} ; 

66: $out ,= $this->month_select($this->name, $datearray) ; 

67: $out .= $this->year select($this->name, $datearray) ; 

68: return $out; 

69: } 

70: 

ri: function day select($fieldname, $datearray) { 

Tas $out = "<select name=\"$fieldname"."['mday']\">\n"; 

fs for ($x=1; $x<=31; $x++) { 

74: $out .= "<option value=\"$x\"". ($datearray[ 'mday' ]==($x) 
75: ?" selected": "").">".sprintf("%@2d", $x) ."</option>\n"; 
76: } 

aT: $out .= "</select>\n'; 

78: return $out; 

F9: } 

80: 

81: function month select($fieldname, $datearray) { 

82: $out = "<select name=\"S$fileldname'’."['mon']\">\n"; 

83: for ($x = 1; $x <= 12; $x++) { 

84: $out .= "<option value=\"".($x)."\"". ($datearrayl[ 'mon' ]==($x) 
85: ?" selected":"")."> ",$this->months[$x-1].°"</option>\n° ; 
86: } 

af: $out .= “</select>\n"; 

88: return $out; 

89: } 

90: 

91: function year select($fieldname, $datearray) { 

92: $out = "<select name=\"$fieldname"."['year']\">"; 

93: $start = $this->getYearStart{}; 

94: $end = $this->getYearEnd(); 

95 : for ($x= $start; $x < $end; $x++) { 

96: $out .= "<option value=\"S$x\"". ($datearray['year' ]==($x) 
97: 7 səlected":"").">",.$x."</option>\n"; 

98: } 

99: Gout ,= "</select>\n"; 

190: return $out; 

191: } 

102: } 


outputO 方 法 丘 据 了 这 些 代 码 的 大 部 分 《第 60 行 到 第 69 行 ) 2 UA 
先 检 查 $timestamp 属 性 ， 除 非 setDate 方 法 中 的 一 个 已 经 调用 了 ， 否 则 


$timestamp 的 值 将 会 设置 为 -1， 并 且 setDate_global0 将 被 默 认 地 调用 。 然 
后 ， 时 间 惟 传递 给 getdate(0) 函 数 来 构建 一 个 日 期 数组 ， 并 且 调 用 方法 来 
产生 每 个 下 拉 列 表 。 


day_select() 方 法 (第 71 行 到 第 79 行 ) 只 是 构建 一 个 HTML select 元 
系 ， 其 中 对 于 一 个 月 中 31 个 可 能 日 期 的 每 一 个 都 有 一 个 option 元 系 。 这 
个 对 和 象 的 当前 日 期 存储 在 $datearray 参 数 变 量 中 ， 在 构建 元 系 设 置 相 关 
的 option 元 素 的 选择 属性 的 时 候 使 用 。sprintfO) 函 数 格式 化 日 期 数 ， 在 日 
期 1 到 9 前 这 加 0。month_select0 和 year_select0) 方 法 〈 第 81 行 到 第 101 行 ) 
使 用 类 似 的 算法 来 构建 月 份 和 年 份 下 拉 列 表 。 


为 什么 我 们 要 把 输出 代码 分 解 为 4 个 方法 ， 而 不 是 简单 地 编写 一 个 
代码 抉 呢 ? 在 构建 一 个 类 的 时 候 ， 有 两 种 程序 员 : 一 种 是 想 要 和 直接 实例 
化 一 个 date_pulldown 对 象 ， 而 男 一 个 想 要 把 date_pulldown 类 子 类 化 以 限 
制 其 功能 。 


对 于 前 者 ， 我 们 想 要 为 类 的 功能 提供 一 个 简单 而 清晰 的 接口 ， 然 后 
这 个 程序 员 可 以 实例 化 一 个 对 象 ， 设 置 其 日 期 ， 并 且 调 用 output() 方 
法 。 对 于 后 者 ， 我 们 想 要 使 得 修改 类 的 功能 的 离 敌 元 系 释 得 容易 。 如 末 
把 所 有 输出 代码 放 入 到 一 个 方法 ， 将 导致 一 个 需要 调整 输出 的 子 类 复制 
大 量 完 全 可 用 的 代码 。 通 过 把 这 些 代 码 分 解 成 离散 的 方法 ， 我 们 束 使 得 
子 关 可 以 改变 有 限 的 部 分 功能 ， 而 不 会 打 乱 整体 秩序 。 例 如 ， 如 采 一 个 
子 关 需要 把 年 份 下 拉 列 表 表 示 为 两 个 单 选 近 钮 ， 程 序 员 只 需要 人 条 单 地 顽 
miyear_select() Ù Yk- 


程序 清单 24.7 包 含 了 一 些 代 码 ， 这 些 代码 调用 库 类 。 在 尝试 执行 这 
些 代 码 之 前 ， 先 取得 来 自 程序 清单 24.6 的 代码 ， 用 PHP 开 始 标记 和 结 


标记 包围 起 来 ， 然 后 保存 到 一 个 名 为 date_pulldown.class.php 的 文件 中 。 


把 这 个 文件 放 到 Web 服 务 器 文档 根 目 录 下 ， 因 为 程序 清早 24.7 将 会 用 到 


它 ， 因 此 它 了 最 好 位 于 那里 。 


人 ”~ 的 Oo Fah c 


ca 


后 ， 
为 ， 


程序 清单 24.7 使 用 date_pulldown 类 


<!DOCTYPE html> 

<html> 

<head> 

<title>Using the date pulldown Class</title> 
</head> 

<?php 

include{ "date pulldown.class.php"); 

$date1 = new date pulldown("fromdate"); 
$datez new date pulldown("todate"}; 


: $date3 = new date pulldown("foundingdate"}; 
: $date3->setYearStart("1972"); 
: if (empty($foundingdate)) { 


$date3->setDate array(array( mday'=>26, ‘mon'=>4, ‘year’ =>1984)}; 


: 】 
: <body> 

: <form> 

: <p><strong>From:</strong><br/> 

: <?php echo $date1->output(}; ?></p> 


: <p><strong>To:</strong><br/> 
: <?php echo $date2->output(}); ?></p> 


: <p><strong>Company Founded:</strong><br/> 
: <?php echo $date3->output(}; ?></p> 


: <button type="submit" name="submit" value="submit ">Submit</button> 
: </form> 
: </body> 
: </html> 


在 第 6 行 ， 我 们 包含 了 date_pulldown.class.php。 包 含 了 类 文件 之 
我 们 可 以 使 用 其 所 有 的 方法 。 对 所 有 下 拉 列 表 都 使 用 类 的 默认 行 
除了 “foundingdate” 以 外 。 对 于 这 个 特殊 的 对 象 ， 我 们 上 履 关 了 默认 的 


开始 年 份 ， 在 第 10 行 将 其 设置 为 1972。 在 第 12 行 ， 我 们 赋 给 这 个 下 拉 列 


表 一 个 任意 的 日 期 ， 这 个 日 期 下 到 表 早 提交 的 时 候 才 会 显示 (参见 图 
24-6) 。 


GES) 


| Using the date_pulldown © 


€ © | © http;//localhost/24/dateselector_withclassphp | 图 K q8 


From: 

[25 [æ] [Mar [=] [2012 |æ] 
To: 
[25 [æ] [Mar [=] [2012 [æ] 
Company Founded: 


'26[e] [Ap [=] [1964 =] 











图 24-6 ”date_pulldown 类 所 产生 的 下 拉 列 表 





Vien: 


这 只 是 一 个 表单 的 前 只， 还 没有 操作 或 方法 ;我们 需要 提供 目 己 的 操作 或 方法 以 便 使 其 
真正 地 做 些 事情 。 


24.3 ”小 结 


在 本 章 中 ， 我 们 把 在 本 书 前 面 学 习 过 的 和 日 期 相关 的 PHP 函 数组 合 
到 一 起 ， 开 发 了 一 个 基本 的 日 历 显 示 应 用 程序 。 我 们 学 习 了 如 何 使 用 
checkdate() 测 试 一 个 输入 日 期 的 有 效 性 ， 并 且 通 过 一 个 示例 脚本 应 用 了 
CAFI — ELR. RASS SERA A ere as A SP 
的 一 种 方法 。 我 们 还 学 习 了 如 何 构 建 一 个 日 期 相关 的 类 库 ， 这 个 类 库 可 
以 用 来 使 得 HTML 表 单 中 操作 日 期 的 示 些 党 玉 工 作 变 得 目 动 化 。 


24.4 Q&A 
Q: 有 很 多 函数 用 来 在 不 同 的 日 历 之 间 转 换 吗 ? 
A: 是 的 。PHP 提 供 了 适合 不 同日 历 的 一 整套 函数 。 我 们 可 以 在 位 


于 http://www.php.net/ manual/en/ref.calendar.php 的 官方 PHP 手 册 中 网 旋 
到 这 些 内 容 。 


24.5 ”实践 练习 


这 个 实践 练习 设计 用 来 帮助 你 预测 可 能 的 问题 、 复 习 已 经 学 过 的 知 
识 ， 并 且 开 始 把 知识 用 于 实践 。 


1. 我 们 使 用 哪个 PHP 函 数 来 创建 一 个 时 间 鹤 ? 


2. 我 们 使 用 哪个 PHP 函 数 来 创建 一 个 和 日 期 相关 的 信息 的 天 联 数 
组 ? 


3. date_pulldown 类 中 的 变量 为 何 要 声明 为 public 的 ? 


oy 


1. mktime() 


2. getdate() 


3. public 的 变量 在 该 类 之 外 也 可 以 使 用 ， 这 些 变 


=I 


里 


要 这 样 。 


EA wel 


1. 修改 日 历 显 示 脚 本 ， 以 显示 整个 一 年 的 日 历 ， 从 1 月 到 12 月 。 然 
后 ， 将 日 历 显 示 为 一 个 3 x 4 的 表格 ， 或 者 4 行 ， 每 行 3 个 月 。 


2. 修改 创建 事件 的 脚本 ， 以 使 用 日 期 下 拉 闫 。 


第 25 章 ”限制 对 应 用 程序 的 访问 


ERER, MÄ FI: 


。 如 何 根据 用 户 、 客 户 机 IP 地 址 、 域 名 和 浏览 器 版 本 来 限制 访问 。 
。 如 何 使 用 Apache 提 供 的 用 户 管理 工具 。 

© 如 何 存 储 和 获取 cookie 信 息 。 

。 如 何 使 用 cookie 进 行 验证 。 


本 章 说 明了 如 何 使 用 Apache， 根 据 用 户 的 且 份 或 请 求 的 信息 来 限制 
对 一 个 站 点 的 访问 。 在 应 用 程序 中 ， 我 们 可 以 创建 自己 的 机 制 来 进行 用 
户 验 证 以 及 通过 cookie 检 查 用 户 的 有 效 性 。 


25.1 ”验证 概览 


授权 和 验证 是 很 多 站 点 经 负 需 要 的 。 首 先 ， 我 们 来 依次 了 解 一 些 定 


验证 建立 了 一 次 通讯 中 的 参与 者 的 号 份 。 我 们 可 以 通过 如 下 方式 来 
获得 验证 : 我 们 所 知道 的 一 个 密码 或 者 一 个 cookie， 或 者 ID 卡 或 铀 证 这 
样 现实 的 东西 ， 或 者 像 指纹 或 虹膜 这 样 我 们 本 喘 所 固有 的 一 部 分 ， 或 者 
是 这 些 元 系 的 任意 组 合 。 在 一 个 站 点 的 环境 中 ， 验 证 人 通常 是 限制 于 使 用 
密码 和 认证 。 

授权 负责 保护 对 资源 的 访问 。 我 们 可 以 根据 几 个 因素 来 授权 访问 ， 
例如 ， 用 户 所 来 目的 卫 地 址 ， 用 户 的 浏 顺 大 类 型 ， 用 户 试图 访问 的 内 容 
或 者 用 户 的 映 份 (在 前 面 通 过 验证 来 定义 )〉。 

Apache 包 含 了 几 个 模块 用 来 提供 验证 和 访问 控制 ， 并 且 它 们 可 以 用 
来 保护 动态 的 和 六 态 的 内 容 。 我 们 可 以 使 用 这 些 模块 中 的 一 个 ， 或 者 在 
应 用 程序 级 实现 目 己 的 访问 控制 并 且 提 供 目 定 义 的 登录 界面 、 单 点 登录 
或 者 其 他 高 级 功能 。 


25.1.1 客户 机 验证 


验证 用 户 是 为 了 记录 或 授权 。HTTP 规 范 提供 了 两 种 验证 机 制 ， 基 
本 验证 和 摘要 验证 。 在 这 两 种 情况 下 ， 过 程 都 是 如 下 上 所 示 。 


1. 客户 机 试图 访问 Web 服 务 器 上 的 限制 内 容 。 


2. Apache A ple aie [—-TS AP BAS. WRIA, 
Apache 返 回 一 个 HTTP 401 状 态 码 ， 表 示 需 要 用 户 验 证 。 


3. 备 户 机 读 取 到 啊 应 并 且 巡 示 用 户 需要 用 户 名 和 密 但 《〈“ 通 利 使 用 
一 个 弹出 对 话 框 ) 。 


4. 客户 机 再 次 尝试 访问 Web 页 面 ， 这 次 ， 把 用 户 名 和 和 密码 作为 
HTTP 请 求 的 一 部 分 传送 。 客 户 机 记 住 用 户 名 和 密码 并 且 在 随后 对 同一 
站 点 的 请 求 中 及 送 它们 ， 这 样 ， 用 户 融 不 需要 对 每 一 个 请 求 都 再 次 输入 
用 户 名 和 密码 了 。 


5. Apache 检 奉 吴 份 的 有 效 性 ， 并 且 根 据 用 户 吴 份 以 及 其 他 的 访问 
规则 来 授权 访问 或 拒绝 访问 。 


在 基本 的 验证 模式 中 ， 用 户 名 和 密码 都 以 明文 传送 ， 作 为 HTTP 请 
求 标 头 的 一 部 分 。 这 融 骏 露 了 一 个 安全 风险 ， 因 为 一 个 攻击 者 可 以 很 容 
TWA BARA aks AD as ZA, RRA AMA, HP A A 
由 地 使 用 它们 。 


摘要 验证 提供 了 更 强 的 安全 性 ， 因 为 它 传送 的 是 一 个 摘要 而 不 十 明 
文 密码 。 摘 要 是 根据 几 个 参数 的 组 合 ， 这 些 参 数 包 括 用 户 名 、 密 人 码 和 请 
求 方法 。 服 务 右 可 以 目 行 计算 摘要 并 检查 知道 宅 码 的 各 己 机， 即便 密 公 
本 号 并 没有 退 过 网 络 传 这 。 





摘要 算法 是 一 种 数学 计算 ， 它 获取 一 个 文本 而 返回 力 一 个 文本 〈 即 摘要 ) ， 摘 要 唯一 地 
标识 了 最 初 的 文本 。 好 的 摘要 算法 应 该 确保 《〈 至 少 为 了 实用 的 目的 ) 不 同 的 输入 文本 产生 不 
同 的 摘要 ， 并 且 无 法 从 摘要 推导 出 最 急 的 输入 文本 。MD5 束 古 一 种 闸 用 的 摘要 算法 。 
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内 部 网 中 。 


在 任何 情况 下 ， 对 于 摘要 验证 和 基本 验证 ， 被 请 求 信息 本 身 未 经 保 
护 地 在 网 络 上 传送 。Web 站 点 的 安全 访问 的 一 个 更 好 的 选择 是 在 SSL 协 
议 上 使 用 HTTP， 我 们 将 在 第 30 章 介绍 。 


25.1.2 用户 管理 方法 


当 验 证 模块 接收 到 来 目 各 户 机 的 用 户 名 和 客人 码 后 ， 它 需要 根据 已 有 
的 用 户 数 据 库 来 验证 其 有 效 性 。 用 户 名 和 获 码 可 以 用 不 同 的 方式 存储 ， 
包括 Apache 所 提供 的 文件 方式 和 基于 数据 库 的 机 制 。 第 三 方 模块 提供 对 
于 像 轻 量 级 目录 访问 协议 (Lightweight Directory Access Protocol, 
LDAP) 和 网 络 信 息 服 务 (Network Information Services, NIS) 等 额外 
机 制 的 文 持 。 


25.2 ”Apache 验证 模块 功能 


Apache 提 供 了 基本 的 框架 和 命令 来 执行 验证 和 访问 控制 。 验 证 模块 
根据 一 个 特定 的 后 端 方法 《文件 、 数 据 库 等 ) 对 验证 密码 提供 文 持 。 用 
户 可 以 可 选 地 组 织 成 组 ， 使 访问 控制 规则 更 容易 管理 。 


Apache 提 供 了 3 种 和 使 用 任何 验证 模块 的 验证 相关 的 、 内 建 的 指 
令 : AuthName、AuthType 和 Require。 


AuthName 接 党 一 个 字符 串 参 数 ， 这 古 验证 领域 的 名 字 。 验 证 领域 
旦 Web 服 务 左上 我 们 可 以 同 其 询问 冤 但 的 一 个 馆 辑 区 域 。 它 将 会 在 浏览 
aa HEY el FAN 
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个 或 多 个 组 名 ， 示 例如 下 。 

Require user joe bob 

或 者 
Reguire group employee contractor 

如 果 我 们 想 要 对 提供 了 一 个 有 效 的 用 户 名 和 密码 的 任何 人 授权 访 
问 ， 我 们 可 以 这 样 做 : 


Require valid-user 


使 用 前 面 的 指令 ， 可 以 控制 谁 能 够 访问 特定 的 虚拟 主机 、 目 录 、 文 
件 等 。 尽 管 验证 和 授权 都 是 独立 的 概念 ， 实 际 上 ， 它 们 在 Apache 中 是 联 
系 在 一 起 的 。 访 问 和 根据 其 体 用 户 吴 份 或 者 组 成 员 来 授权 。 一 些 第 三 方 模 
块 ， 例 如 ， 茶 些 基 于 LDAP 的 模块 ， 在 验证 和 授权 之 间 进 行 了 清晰 的 划 


ne 


包含 在 Apache 中 的 验证 模块 提供 了 如 下 功能 。 


。 后 闪存 储 一 一 捉 供 了 包 人 用 户 名 和 组 信息 的 文本 或 数据 库 文件 。 
。 用 户 管理 一 一 近 供 了 在 后 站 存储 中 创建 和 管理 用 户 和 组 的 工具 。 
。 授权 信息 指定 模块 的 结 来 是 否 是 授权 的 。 
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中 找到 ， 或 者 因为 没有 匹配 他 们 的 信息 的 验证 规则 。 在 这 个 例子 中 ， 将 会 发 生 如 下 两 种 情况 
A 


。 如 采 模 块 指 定 目 己 的 结果 为 授权 ， 用 户 将 会 被 拒绝 访问 ， 并 且 Apache 将 会 返回 一 个 错 
Ro 


。 如 采 模 块 指 定 目 己 的 结果 为 没有 授权 ， 其 他 借 块 可 以 有 一 个 验证 该 用 户 的 机 会 。 





这 就 使 得 我 们 能 够 有 一 个 主 授权 模块 ， 它 知道 大 多 数 用 户 ， 并 且 能 够 拥有 其 他 的 授权 模 
块 ， 这 些 便 块 可 以 验证 其 他 的 用 户 。 


25.2.1 基于 文件 的 验证 


mod_auth Apache 模 块 提供 了 通过 包含 用 户 名 和 密码 的 文本 文件 进 
行 的 基本 验证 ， 类 似 于 传统 的 UNIX 验 证 如 何 使 用 /etc/passwd 
和 /etc/groups 文 件 。 


1. 后 合 存 储 

当 我 们 使 用 后 台 存 储 方 法 ， 你 需要 指定 包含 了 用 户 名 和 密码 的 列表 
的 文件 ， 并 且 这 个 文件 可 选 地 包含 了 组 的 列表 。 

这 个 用 户 文 件 是 一 个 UNIX 取 的 密码 文件 ， 包 含 了 用 户 的 名 字 和 加 
短 的 密码 。 在 UNIX 上 ， 使 用 了 加 密 算 法 后 ， 整 个 看 上 去 如 下 所 示 。 
admin: iFrlxqg@Q6RQ6 

在 Windows 系 统 ， 使 用 MD5 算 法 ， 如 下 上 所 示 。 
admin: $apri$Ug3..... $jVTedbQWwBKTfXsn5j K6UX/ 

组 文件 包含 了 一 个 组 以 及 属于 每 个 组 的 用 户 的 列表 ， 中 间 用 空格 隔 
F. RPM BA 
web: admin joe Daniel 

AuthUserFile 和 AuthGroupFile 指 令 接 受 一 个 路 径 参 数 ， 访 参数 指 同 
用 户 文件 和 组 文件 。 组 文件 是 可 选 的 。 
2. HP ee 

Apache 发 布 版 包括 了 UNIX 上 的 htpasswd 工 具 以 及 Windows 上 的 
htpasswd.exe， 它 们 都 设计 用 来 帮助 我 们 演 理 用 户 密 码 文 件 。 两 个 版 本 
的 功能 相同 ， 但 是 Windows 版 本 使 用 一 种 不 同 的 方法 来 加 和 窒 密 码 。 加 密 
对 于 用 户 和 管理 员 是 透明 的 。 


在 LinuxUNIX 上 ， 第 一 次 深 加 一 个 用 户 的 时 候 ， 我 们 需要 输入 如 下 
A 人 
X o 


a 


fusr/locaL/apache2/bin/htpasswd -c file userid 


这 里 fle 是 包含 用 户 名 和 和 密码 列表 的 文件 ， 而 userid 则 是 要 添加 的 用 
户 名 。 将 会 提示 我 们 输入 一 个 密码 ， 并 且 将 会 创建 文件 。 例 如 ， 在 
Linux/UNIX， 执 行 如 下 命令 。 


fusr/local/apache2/bin/htpasswd -c /usr/local/apache2/conf/htusers admin 


上 述 命令 创建 了 密码 文件 /usr/local/apache2/conf/htusers， 并 且 添 加 
J admin F . 


关 似 的 功能 在 Windows 上 也 存在 ， 这 里 的 命令 行 操作 看 上 去 如 下 所 
不 。 


htpasswd -c "C:\Program Files\Apache Software Foundation\Apache2.2\conf\htusers' 
admin 


-MSTA E VEhtpasswd AY AIT CHF, “EIA AEC. SER 
AA EEE Ss Bl CE SCE, ANE E-card, AAMC 
(F A) BE AS ath o 
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3. 使 用 mod auth 

程序 清单 25.1 给 出 了 一 个 配置 示例 ， 它 限制 htusers 密 码 文 件 中 的 授 
权 用 户 访 问 文 档 根 目录 下 的 私有 目 孙 。 注 意 ， 可 选 的 AuthGroupFile 指 令 
并 没有 给 出 。 


程序 清单 25.1 基于 文件 的 验证 示例 


: <Directory /usr/local/apache2/htdocs/private> 
: AuthType Basic 

: AuthName "Private Area" 

: AuthUserFile /usr/local/apache2/conf /htusers 
: Require valid-user 

: «</Directory> 


Om bw hy — 


25.2.2 ”基于 数据 库 文 件 的 访问 控制 


以 纯 文 本 文件 存储 用 户 名 和 密码 非常 方便 ， 但 是 这 种 方法 缺乏 灵活 
性 。Apache 将 需要 顺序 地 打开 并 读 取 文件 ， 以 查找 某 个 特定 用 户 。 当 用 
户 的 数量 增加 ， 这 一 操作 将 变 得 非常 耗 时 。mod_auth_dbm 模 块 使 得 我 
们 可 以 用 索引 的 数据 库 文件 来 蔡 代 基于 文本 的 文件 ， 这 样 可 以 处 理 更 多 
数量 的 用 户 而 不 会 导致 性 能 下 降 。mod_auth_dbm 模 块 包含 在 Apache 
中 ， 但 默认 不 可 用 。 当 构建 Apache 配 置 的 时 候 ， 使 用 --enable- 
module=dbm 选 项 就 可 以 打开 这 一 模块 。 


1， 后 端 存储 
mod_auth_dbm 模 块 提 供 了 两 个 指令 ，AuthDBMUserFile 和 和 

AuthDBMGroupFile， 它 们 指 同 了 包含 用 户 名 和 组 的 数据 库 文 件 。 和 纯 

文本 文件 不 同 ， 这 两 条 指令 可 以 指 癌 同一 个 文件 ， 其 中 既 包 含 用 户 也 包 

| 

2. HPEH 


Apache 提 供 了 一 种 UNIX 和 Windows 用 户 都 可 以 使 用 的 工具 ， 叫 做 
htdbm， 和 所 允许 你 创建 和 管理 存储 在 数据 库 文 件 中 的 用 户 和 组 。 可 以 在 
Apache 发 布 的 bin 目 录 下 找到 这 个 工具 ， 一旦 找到 了 ， 输 入 如 下 内 容 ， 
ly Br a Fe US IS FA Pe 


htdbm -c databasename userid 


然后 会 提示 你 输入 密码 ， 并 且 将 该 用 户 添加 到 新 的 数据 库 文 件 中 。 
如 果 你 需要 删除 用 户 daniel， 可 以 执行 如 下 的 命令 。 


htdbm -x databasename daniel 


可 以 通过 执行 不 带 任何 参数 的 htdbm， 从 而 得 到 该 命令 的 完整 的 语 
法 信息 。 


25.3 ”使 用 Apache 进 行 访问 控制 


mod_access 模 块 默认 可 用 ， 人 允许 我 们 根据 客户 机 请 求 的 参数 ， 例 如 
一 个 具体 标 头 、 卫 地 址 或 客户 机 的 主机 名 的 出 现 ， 来 限制 对 资源 的 访 
问 。 
25.3.1 ”实现 访问 规则 

我 们 可 以 使 用 Allow 和 Deny 指 令 来 指定 访问 规则 。 这 两 个 指令 都 接 
受 IP 地 址 、 环 境 变 量 和 域名 这 样 的 一 个 参数 列表 。 
1. 通过 IP 地 址 允许 /拒绝 访问 

我 们 可 以 根据 客户 机 的 卫 地 址 来 拒绝 或 授权 访问 ， 示 例如 下 。 
Allow from 108.0.9.1 109.0.0.2 18.0,0.3 


RATE DA 7B o> ES PH ok Bk ee — 7S Dd 28 FS OR E KE TP HE OE A) 
YUH. Fab, RII Uta EPERE “N.S PSPS PT. E 
含 这 些 的 任何 IP 地 址 部 符合 这 一 规则 ， 示 例如 下 。 


Deny from 18.9 


上 述 规则 将 会 匹配 所 有 以 10.0 打 头 的 地 址 ， 如 10.0.1.0 和 10.0.0.1。 
我 们 也 可 以 利用 卫 地 址 和 网 络 掩 码 ， 卫 地 址 指定 了 网 络 ， 而 捧 码 指 
定 了 哪些 位 属于 网 络 前 级 而 哪些 位 属于 节点 ， 示 例如 下 。 
Allow from 180.0.0.0/255.255.255.@ 


将 会 匹配 卫 地 址 10.0.0.1、10.0.0.2 一 直到 10.0.0.254。 


RATH RTE oo se a DY Le RT FE PR ES. PO, RAI 
可 以 把 上 述 的 规则 改写 如 下 。 
Allow from 10.80.0.0/24 


2. 通过 域名 允许 /拒绝 访问 


我 们 可 以 根据 指定 主机 名 或 者 部 分 域名 来 控制 访问 。 例 如 ，Allow 


from example.com 将 会 匹配 www.example.com，foo.example.com 等 。 


提示 : 





根据 域名 打开 访问 规则 过 使 Apache 在 客户 机 地 址 上 反问 查找 DNS， 而 忽略 挥 
HostNameLookups 指 令 的 设置 。 这 个 过 程 是 隐 式 地 执行 的 。 


3. 根据 环境 变量 允许 /拒绝 访问 


你 可 以 根据 出 现 东 个 坏 境 变 量 来 指定 访问 规则 ， 通 过 在 变量 名 的 前 
面 加 上 和 字符 串 env= 前 级 。 我 们 可 以 使 用 这 一 功能 来 授权 或 拒绝 访问 某 个 
浏 哆 颖 或 者 浏览 絮 版 本 ， 以 防止 菜 个 站 点 链接 到 资源 等 等 。 为 了 让 这 个 
例子 像 期 竺 的 那样 工作 ， 客 良机 需要 传送 User-Agent 标 凑 ， 示 例如 下 。 


BrowserMatch MSIE il1explorer 
Deny from env=iexplorer 


HT PL AIA Sf User-Agent k, EA A HER GMS Be ERIE , 
但 征 大 多 数 用 户 不 会 这 么 做 ， 并 且 这 一 技术 在 大 多 数 情 况 下 征 有 效 的 。 


4. 允许 /拒绝 所 有 客户 机 访问 


天 键 字 all 匹 配 所 有 客户 机 。 我 们 可 以 指定 Allow from all 或 Deny 
from all 来 授权 或 拒绝 所 有 客户 机 访问 。 


25.3.2 ”应 用 访问 规则 


我 们 可 以 有 效 个 Allow 和 Deny 访 问 规 则 。 我 们 可 以 通过 Order 指 令 来 
选择 应 用 访问 规则 的 顺序 。 后 应 用 的 规则 共有 更 高 的 优先 级 。 规 则 接收 
一 个 参数 ， 这 个 参数 可 以 是 Deny，Allow、Allow，Deny 或 Mutual- 
Failure。Deny，Allow 古 Order 指 令 的 默认 值 。 注 意 ， 值 中 没有 空格 。 


1. Deny, Allow 


Deny，Allow 指 定 了 Deny 命 令 是 在 Allow 指 令 之 前 应 用 的 。 通 过 
Deny，Allow， 如 果 没 有 Allow 或 Deny 指 令 或 者 客户 机 没有 匹配 到 任何 
规则 ， 客 户 机 默认 被 授 权 访 问 。 如 采 客 户 机 匹配 一 条 Deny 规 则 ， 它 将 入 
拒绝 访问 ， 除 非 它 也 匹配 了 一 条 Allow 规 则 ， 该 Allow 规 则 具有 优先 权 ， 
因为 Allow 指 令 在 后 面 应 用 ， 并 且 有 具有 更 高 的 优先 级 。 


程序 清单 25.2 展 示 了 如 何 配 置 Apache 以 允许 来 目 内 部 网 络 或 者 域 
example.com 的 客户 机 访问 /Private 位置， 而 拒绝 其 他 的 任何 人 访问 。 


程序 清单 25.2 ”示例 Deny，Allow 访 问 控 制 配置 


: sLocation /private> 

Order Deny ,Allow 

Deny from all 

Allow from 1@.0.0.0/255.255.255.@ example.com 
: </Location> 


me ù V 一 


2. Allow, Deny 


Allow，Deny 指 定 了 Allow 指 令 在 Deny 指 令 之 前 应 用 。 通 过 
Allow,Deny， 如 果 没 有 Allow 或 Deny 指 令 或 者 如 果 客 户 机 没有 匹配 任何 
规则 ， 客 户 机 默认 被 拒绝 访问 。 如 采 客 户 机 匹配 任何 Allow 规 则 ， 将 会 
允许 它 访 问 ， 除 非 它 也 匹配 了 一 条 Deny 规 则 ，PDeny 规 则 将 具有 优先 


仅 。 


注意 ， 不 带 有 任何 Allow 或 Deny 规 则 的 Order Allow，Deny 的 出 现 ， 
会 叶 任 对 某 个 资源 的 所 有 请 求 锐 拒绝 ， 因 为 默认 的 行为 是 拒绝 访问 。 


程序 清单 25.3 允 许 访问 一 个 特定 主机 之 外 的 任何 主机 。 


程序 清单 25.3 ”示例 Allow,Deny 访 问 控制 配置 


<Location /Somey/ location/> 
Order Allow, Deny 
Allow from all 


Deny from host.example.com 
</Location> 


cm S one 一 


3. Mutual-Failure 


在 Mutual-Failure 的 情况 下 ， 只 有 当主 机 [匹配 一 条 Allow 指 令 并 日 不 
匹配 任何 的 Deny 指 令 的 时 候 ， 才 授权 主机 访问 。 


25.4 组 合 Apache 访 问 方 法 


在 上 一 节 中 ， 我 们 芝 习 了 如 何 根据 用 户 喘 份 或 请 求 信 息 来 限制 访 
问 。Satisfy 指 令 人 允许 我 们 确定 ， 为 了 授权 访问 是 人 否 两 种 访问 类 型 限制 必 
须 按 顺序 满足 。S$atisfy 接 收 一 个 参数 ， 要 么 是 all， 要 么 是 any。 


Satisfy ERE MRAP NLE S NA AH A AA a HA 
过 了 访问 限制 ， 客 户 机 将 被 授 权 访 问 。Satisfy anay ERE WAR EP NL E 
供 一 个 有 效 的 用 户 名 和 蜜 但 或 者 通过 了 访问 限制 ， 和 客户 机 将 被 授权 访 
问 。 


为 什么 这 条 指令 有 用 呢 ? 例如， 你 可 能 想 要 为 来 日 内 部 的 、 可 信 的 
地 址 的 用 户 提 供 对 Web 站 点 的 目 由 访问 ， 但 需要 那些 来 日 Internet 的 用 户 
提供 一 个 有 效 的 用 户 名 和 密码 。 程 序 清单 25.4 展 示 了 这 一 点 。 


程序 清单 25.4 ”混合 的 验证 和 访问 控制 规则 


: <Location /restricted> 

: Allow from 180.0.@.@/255.255.255.0 

: AuthType Basic 

: AuthName "Intranet" 

: AuthUserFile /usr/local/apache2/conf /htusers 
: Require valid-user 

: Satisfy any 

: </Location> 


O oa kh wh — 








根据 连接 或 请 求 信息 的 访问 控制 并 不 完全 安全 。 尺 官 它 在 大 多 数 情 况 提 供 了 一 个 相应 级 
列 的 保护 ， 但 规则 依赖 于 你 的 DNS 服 务 费 的 完整 性 和 网 络 基 础 设施 。 如 末 一 个 攻击 者 获取 了 
对 DNS 服 务 占 的 控制 ， 或 者 路 由 器 或 防火 墙 配 置 不 正确 ， 他 可 以 轻松 地 把 授权 域名 记录 修改 
为 指 问 他 目 己 的 机 帮 ， 或 者 假 容 他 来 目 一 个 授权 的 IP 地 址 。 


25.5 ”根据 HTTP 方 法 限制 访问 


通常 ， 我 们 和 希望 访问 控制 指令 应 用 于 所 有 类 型 的 客户 请 求 ， 并 且 这 
是 一 个 默认 的 行为 。 然 而 ， 在 某 些 情况 下 ， 我 们 只 想 对 某 些 HTTP 方 
法 ， 如 GET 和 HEAD， 应 用 验证 和 访问 规则 。 


<Limit> 容 乾 接 受 方法 的 一 个 列 琢 ， 并 且 包 侣 了 指令 ， 这 些 指令 应 
用 于 包含 那些 方法 的 请 求 。 可 用 的 方法 的 完整 的 列表 是 : GET, 
POST, PUT, DELETE, CONNECT, OPTIONS, TRACE, PATCH, 
PROPFIND,. PROPPATCH, MKCOL, COPY, MOVE, LOCK! 
UNLOCK. 


<LimitExcept> 段 供 了 一 个 补充 功能 ， 它 所 包含 的 命令 应 用 于 那些 不 
包含 所 列 出 的 方法 的 请 求 。 


程序 消音 25.5 给 出 了 默认 的 Apache 配 置 文件 的 一 个 例子 。<Limit> 和 
<LimitExcept> 段 允许 只 读 的 方法 ， 但 是 拒绝 对 任何 可 以 修改 文件 系统 内 
容 的 其 他 方法 (如 PUT 方 法 ) 的 请 求 。 要 了 解 有 天 这 里 可 用 的 多 种 选项 
的 更 多 信息 ， 请 参考 位 于 http://httpd.apache.org/docs-2.2/ mod/core.html 
的 Apache 文 档 。 


程序 清单 25.5 ”根据 规则 限制 访问 


1: <Directory /home/*/public html> 

2: AllowOverride FileInfo AuthContig Limit 

3: Options MultiViews Indexes SymLinksIfOwnerMatch IncludesNoExec 
4: «Limit GET POST OPTIONS PROPFIND> 

ae Order Allow,Deny 

6: Allow from all 

7: </Limit> 

8: <LimitExcept GET POST OPTIONS PROPFIND> 

9: Order Deny,Allow 


18: Deny from all 
11: </LimitExcept> 
12: </Directory> 


在 下 一 节 中 ， 我 们 还 将 要 学 习 根 据 cookie 中 的 信息 在 应 用 程序 中 限 
制 访问 。 


25.6 ”根据 cookie 值 限制 访问 


在 第 12 章 ， 我 们 学 习 了 cookie 的 所 有 结构 ， 以 及 在 PHP 中 如 何 设置 
和 访问 cookie 变 量 。 下 面 几 贡 将 介绍 cookie 在 验证 过 程 中 的 一 些 实 际 应 
FA. 


假设 你 已 经 创建 了 一 个 登录 表单 ， 用 来 根据 数据 库 检 查 值 。 如 果 用 
尸 是 授权 的 ， 我 们 发 送 一 个 cookie 来 说 明 这 一 点 。 随 后 ， 对 于 仪 限 于 授 
权 的 用 户 访 问 的 所 有 页 面 ， 我 们 检查 这 个 其 体 的 cookie。 如 条 存 在 
cookie， 用 户 可 以 到 看 页 面 。 如 条 不 存在 cookie， 用 户 要 么 达 回 到 登录 
表意， 要么 及 送 一 条 显示 于 屏 医 上 的 基于 访问 限制 的 消息 。 我 们 将 在 下 
面 的 小 节 中 经 历 这 些 步骤 。 


25.6.1 创建 授权 用 户 表 


当 我 们 要 把 用 户 账 尸 集 成 到 一 个 基于 Web 的 应 用 程序 中 时 ， 把 与 用 
己 相 关 的 信息 存储 到 一 个 数据 库 表 中 ， 这 十 第 见 的 做 法 。 随 后 ， 这 个 表 
中 的 信息 可 以 用 来 授权 用 户 并 且 授 予 这 些 “ 特 殊 ” 用 户 对 专门 的 站 乓 区 域 
的 访问 。 


如 下 的 表 创 建 命令 将 会 在 你 的 MySQL 数 据 库 中 创建 一 个 名 为 
auth_users 的 表 ， 其 中 具有 ID、 和 名字、 姓氏 、Email 地 址 、 用 户 名 和 密 伍 
的 字段 。 


CREATE TABLE auth_users ( 
id int NOT NULL PRIMARY KEY AUTO INCREMENT, 
f name VARCHAR (50), 
l name VARCHAR (56), 
email VARCHAR(150), 
username VARCHAR(25), 
password VARCHAR({41) 


如 下 的 INSERT 命 令 在 auth_users table 表 中 为 名 为 John Doe 的 用 户 插 
入 了 一 条 记录 ， 其 Email 地 址 为 john@doe.com， 用 户 名 为 jdoe， 而 密码 
为 doepass。 


INSERT INTO auth users VALUES ('1', ‘John’, ‘Doe’, ‘john@doe.com', 
'jdoe', PASSWORD( '‘doepass')); 


这 条 INSERT 命 令 应 该 是 一 目 了 然 的 ， 除 了 PASSWORDO0O 函 数 的 用 
法 。 当 这 个 函数 用 在 INSERT 命 令 中 时 ， 存 储 在 表 中 的 实际 上 不 是 真正 
的 密码 ， 而 是 密码 的 一 个 41 字 符 的 哈 希 值 。 


当 我 们 查看 auth_users 表 的 内 容 的 时 候 ， 我 们 会 在 passworld 字 上段 中 
看 到 哈 希 值 ， 如 下 所 示 。 


再 二 下 竹本 二 SORES ts ee eater ores SSSA Meera Aa E + 
| username | password | 
a ana i + 
| jdoe | *@AAD744979343D58A7F17A5@E514E6AD6533D04B | 
下 ee ermede Poss geo. EEE RES SEE cues eee eye, eer 十 


尽 过 它 看 上 去 像 是 加 密 的 ， 哈 希 值 实际 上 不 是 信息 的 一 个 加 密 数 
扼 。 实 际 上 ， 它 是 原始 信息 的 一 个 “指纹 ”。 哈 布 值 通 利 用 来 执行 匹配 ， 
束 像 指纹 一 样 。 在 这 个 例子 中 ， 当 你 要 检查 目 己 的 用 户 密码 的 时 候 ， 将 
便 用 输入 的 哈 希 值 来 匹配 存储 的 哈 硕 值 。 使 用 哈 希 值 避免 了 存储 实际 窗 
但， 也 避免 了 安全 风险 。 


25.6.2 ”创建 登录 表单 和 脚本 


TERM SPINAL ZJE, RIIT Be — Pb OR AE Ff 


FERS BIS A, ce“ TR A BCH, URE H25.6 ATA . 


owl aor why — 


oo 


程序 清单 25.6 用户 登录 页 面 


<!DOCTYPE html> 

<html> 

<head> 

<title>User Login Form</title> 

</heaa> 

<body> 

<hi>Login Form</h1> 

<form method="post" action="userlogin.ohp"> 

<p><label for="Uusername'><strong=username: </strong><br/> 


: <input type="text" id="username’ name="username’ /></Llabel></p> 

: <p><label for="password'><strong=password:</strong><br/> 

: <input type="password" id="password" name="password" /></label></p> 
: <button type="submit" name="submit" value="login">Login</button= 

: </form> 

: </body> 

: </html> 


把 这 些 代码 放 入 到 一 个 名 为 loginform.html 的 文本 文件 中 ， 并 且 将 这 


个 文件 放置 到 Web 服 务 器 文档 根 目 录 下 。 接 下 来 ， 我 们 将 创建 脚本 本 
导 ， 表 单 脚本 名 为 userlogin.php 〈 人 参见 程序 清单 25.7) 。 


O wy a a oh — 


to 


程序 清单 25.7 用 户 登 录 脚 本 


<?php 

¿icheck for required fields from the form 

if (({!isset($ POST['username'])}) |] {!isset($ POST['password']})) 1 
header(*Location: userlogin.html"); 
exit; 

} 

/fconnect to server and select database 

Smysqli = mysqli connect("localhost", "“joeuser", "somepass", "testDB") 


or die({mysqli error()}; 


/fuse mysqli_real_escape string to clean the input 


: $username = mysqli real escape string($mysqli, $ POST[ 'username']}; 
: $password = mysqli real escape string($mysqli, $ POST[ password']}; 


15: 
18; 
Ve: 
18; 
19: 
od: 
2l; 
22 : 
eo} 
24: 
20% 
26: 
of: 
og: 
29; 
30: 
aT: 
32: 
3a: 
34: 
35: 
36: 
oft 
383 
39} 
40: 
41: 
42: 
43: 
44; 
45: 
AG: 
4r: 
48: 
AQ: 
5o: 
ifs 
52i 
Ba: 
54} 
55: 
56: 
of: 


:icreate and issue the query 
$sql = "SELECT f name, 1 name FROM auth users WHERE 
username = '",.$username."' AND 
password = PASSWORD('".$password."')"; 
$result = mysqli _query(Smysqli, $sql} or die(mysqli_error($mysqli}}; 


iget the number of rows in the result set; should be 1 if a match 
if (mysqli_num_rows{$result) == 1) { 


:iif authorized, get the values of f_name 1 name 
while ($info = mysqli fetch array($result)) { 

$f name stripslashes({$info['f_name']); 

$1 name = stripslashes{$info['l name']); 


i 


{iset authorization cookie 
setcookie("auth’, "1", @, "Z", "yourdomain.com", @); 


ficreate display string 
$display block = " 


<p>" .$f name." ",$l name." is authorized!</p> 
<o>Authorized Users' Menu: </p> 
< |) ]> 
<li><a href=\"secretpage.php\ >secret pagqe</a></li> 
</ul>"; 

t else 1 


/fredirect back to login form if not authorized 
header({ "Location: userlogin.html"}; 
exit: 

} 

¿iclose connection to MySQL 

mysqli close($mysqli); 

?> 

<!DOCTYPE html> 

<html> 

<head> 

<title>User Login</title> 

</head> 

<body> 

<?php echo $display block; ?> 

</body> 

</html> 


把 上 述 代码 放 入 到 一 个 名 为 userlogin.php 的 文本 文件 中 ， 在 第 32 行 


把 “your-domain.com” 修 改 为 你 实际 的 域名 ， 并 且 把 该 文件 放置 到 Web 服 
务 堪 文档 根 目 录 下 。 此 时 ， 你 可 以 答 斌 进行 一 下 ， 但 先 来 看 看 代 但 将 做 
eT Ae 


PATRAN ae EE, ThE PIM A PE: 
$_POST[username] 和 $_POST['password]。 如 果 这 些 字 段 中 的 任何 一 个 
不 存在 ， 脚 本 都 把 用 户 重 定 问 到 最 初 的 登录 表单 。 如 果 两 个 字段 部 存 
在 ， 脚 本 跳 到 第 9 行 ， 从 那里 连接 到 数据 库 服务 器 ， 以 准备 执行 SQL 得 
询 来 检查 用 户 的 号 份 。 在 执行 查询 之 前 ， 第 13 行 和 第 14 行 将 用 户 输 入 的 
数据 安全 化 。 一 旦 信息 安全 化 了 ， 执 行 但 询 ， 可 以 在 第 17 行 到 第 20 行 找 
到 。 注 意 ， 但 询 根 据 存 储 在 表 中 的 密码 ， 检 查 了 来 目 表 单 输入 的 密码 的 
het 40 TEL» 


XNR DA LAC, SF AAT AY RIN AA, tite iM 
授权 的 用 户 。 


第 23 行 通过 计算 结果 集中 有 的 行 数 来 测试 查询 的 结果 。 如 果 用 户 名 和 
密码 对 代表 一 个 有 效 的 登录 ， 行 数 计 算 应 该 确切 地 为 1。 如 果 是 这 种 情 
况 ， 在 第 26 行 到 第 29 行 ，mysqli_fetch_array() 函 数 用 来 提取 用 户 的 名 字 
和 姓氏 。 这 些 名 字 只 是 为 了 美观 。 


第 32 行 设置 了 授权 cookie， 这 个 cookie 的 名 字 是 auth， 并 且 值 为 1。 
如 采 在 这 个 时 间 衬 档 放 入 一 个 0， 只 要 这 个 用 户 的 web 浏 虎 需 会 话 打 开 
看 ，cookie 融 会 持续 。 当 用 户头 财 了 浏 虎 匀 ，cookie 将 过 期 。 第 35 行 到 
第 40 行 创建 了 一 条 供 显示 的 消息 ， 包 括 到 一 个 文件 的 链接 ， 我 们 稍 后 将 
创建 这 个 文件 。 最 后 ， 在 第 41 行 到 第 45 行 处 理 一 个 失效 的 登录 符 试 。 在 
这 个 例子 中 ， 用 户 锌 直接 章 定 同 到 最 初 的 登录 表 早 。 


访问 登录 表单 ， 并 且 为 John Doe 用 户 输入 有 效 的 值 。 当 你 提交 了 表 
单 ， 结 果 如 图 25-1 所 示 。 





[a] User Login 


< C | 人 @@ http://localhost/25/userlogin.php x, 


John Doe is authorized! 
Authorized Users’ Menu: 


è secret page 


图 25-1 ”成功 的 登录 结果 


和 试用 一 个 无 效 的 用 户 名 和 黎 码 对 登录 ， 你 应 诅 被 重 定 同 到 登录 表 
单 。 在 下 一 下 《也 是 最 后 一 节 ) 中 ， 我 们 将 创建 secretpage.php 脚 本 ， 它 
将 读 取 我 们 刚 设 置 的 验证 cookie 并 进行 相应 的 操作 。 

25.6.3 jliz\auth cookie 


这 个 问题 的 最 后 一 部 分 就 是 使 用 auth cookie 的 值 ， 从 而 允许 用 户 访 
问 一 个 私有 文件 。 在 这 个 例子 中 ， 所 涉及 的 文件 在 程序 清单 25.8 中 给 
出 。 


程序 清单 25.8 ”检查 auth cookie 


1: <?php 

2: if ($ COOKIE['auth'] == "1") { 

3: $display_block = "<p>You are an authorized user.</p>"; 
A: } else { 

sie f/fredirect back to login form if not authorized 
6: header("Location: userlogin. html"); 

ft exit; 

8: } 

9, ?> 

18: <!DOCTYPE html> 

11: <html> 

12: <head> 

13: <title>Secret Page</title> 

14: </head> 

15: <body> 

16: <?php echo $display block; 7> 

17: </body> 

18: </html> 


在 图 25-1 所 示 的 菜单 中 ， 点 击 secret page 链 接 。 由 于 你 是 一 个 被 授 
权 的 用 己 ， 应 访 看 到 如 图 25-2 所 示 的 结 


= 


[a] secret Page =- 


€ C | ®© http://localhost/25/secretpage.php A, 


You are an authorized user. 


图 25-2 ”作为 一 个 授权 用 户 访 问 secret page 


关闭 浏览 器 并 且 尝 试 直接 访问 secretpage.php。 你 将 会 发 现 ， 无 法 直 
接 访问 ， 并 且 将 会 午 定 同 到 最 和 初 的 登录 页 面 ， 因 为 验证 cookie 还 没有 设 
置 。 


25.7 小结 


本 章 介 绍 了 如 何 使 用 Apache 功 能 来 根据 远程 用 户 的 身份 ， 以 及 来 目 
HTTP 请 求 或 网 络 连接 的 信息 限制 对 Web 站 点 的 访问 。 本 章 还 介绍 了 一 
些 验 证 模块 ， 包 括 Apache 和 其 他 工具 ， 这 些 模块 和 工具 可 以 用 来 创建 并 
党 理 用 户 和 组 数据 库 。 


此 外 ， 我 们 学 习 了 使 用 cookie 值 来 多 诗 访 问 PHP 应 用 程序 的 特定 音 
分 的 一 种 方法 。 


25.8 Q&A 


Q: 我 已 经 有 一 个 UNIX 系 统 ， 可 以 使 用 /etcpasswd 作 为 用 户 
数据 库 吗 ? 


A: 尺 害 使 用 /etc/passwd 可 能 看 上 去 很 方便 ， 但 还 是 建议 你 不 要 使 
用 已 有 的 /etc/passwd 文 件 来 验证 站 点 的 有 用户。 否则 ， 获 取 了 对 Web 丫 点 
的 用 户 的 访问 权限 的 攻击 者 ， 将 能 够 获取 对 系统 的 访问 权限 。 体 持 数 据 
库 隔 离 并 且 鼓 励 用 户 为 系统 账户 和 Web 访 问 设置 不 同 的 密码 。 对 于 安全 
性 较 弱 的 答 码 以 及 用 户 名 了 驶 是 密码 的 账户 ， 定 期 运行 竹 码 检 奏 需 来 扫 摘 


它们 。 
Q: 为 什么 在 某 些 Web 站 点 中 需要 两 次 密码 ? 


A: 浏览 器 记录 了 和 警 码 ， 以 便 我 们 不 用 在 每 次 请 求 的 时 候 都 必须 输 
入 密码 。 存 储 的 密码 基于 领域 (AuthName 命 令 ) 以 及 Web 站 点 的 主机 
名 。 有 上 时候， 我 们 可 以 通过 不 同 的 名 字 《〈 如 yourdomain.com 和 
www.yourdomain.com) 来 访问 一 个 Web 站 上 点。 如果 授权 你 访问 
yourdomain.com 的 示 一 个 限制 区 域 ， 但 被 重 定 回 或 者 打开 到 
www.yourdomain.com 的 链接 ， 将 会 再 次 要 求 提 供用 户 名 和 密码 ， 因 为 
A ae UA Ak cE PSE |B] A Web itt K o 


Q: cookie 是否 会 引起 任何 严重 的 安全 或 隐私 问题 ? 


A: Hk aii [A] cookie H Ree A OREN. SS 
cookie") 以 存储 在 用 户 的 便签 上 ， 但 没有 对 用 户 文件 系统 的 其 他 访问 权 
限 。 然 而 ， 有 可 能 设置 一 个 cookie 来 啊 应 对 一 个 图 像 的 请 求 。 因 此 ， 如 
末 很 多 站 点 包含 了 一 个 第 三 方 广告 服务 套 或 计数 硕 脚 本 所 提供 的 网 像 ， 
第 三 方 可 能 能 够 路 越 多 个 域 来 跟 踩 一 个 用 户 。 


25.9 KRAJ 


这 个 实践 练习 设计 用 来 帮助 你 预测 可 能 的 问题 、 复 习 已 经 学 过 的 知 
识 ， 并 且 开 始 把 知识 用 于 实践 。 


1. 相对 于 纯 文 本 文件 ， 用 来 存储 用 户 验 证 信息 的 数据 库 文 件 的 优 


势 是 什么 ? 
2. 能 否 说 出 HITP 基 本 验证 的 一 些 缺 点 ? 


3， 哪 个 函数 设计 用 来 允许 我 们 在 访问 者 的 浏览 磺 上 设置 一 个 


cookie? 


oy 


fi E 


1. 数据 库 文 件 具 有 更 好 的 可 扩展 性 ， 因 为 它们 可 以 索引 。 这 意味 
者 Apache 不 需要 顺序 读 取 文件 直到 找到 针对 一 个 特定 用 户 的 匹配 ， 而 是 
可 以 跳 到 准确 的 位 置 。 


2. 一 个 缺点 是 ， 信 息 在 网 络 上 以 明文 传输 。 这 意味 看 ， 除 非 你 使 
用 SSL， 奋 则 攻击 者 有 可 能 读 取 到 浏览 器 发 送 到 服务 器 的 数据 包 并 且 和 饭 
取 你 的 密码 。 另 一 个 缺点 是 ，HTTP 验 证 不 会 提供 一 个 工具 来 自 定 义 登 
K (除了 领域 名 ) 。Web 站 点 使 用 HTML 表 单 和 cookie 来 实现 自 定义 登 
录 机 制 是 很 常见 的 。 


3. setcookie() 隙 数 允许 你 设置 一 个 cookie， 尽 管 你 也 可 能 使 用 
header() K 2447 HH —7Set Cookie 标 头 。 


EA wel 


练习 在 你 的 开发 服务 需 上 使 用 不 同类 型 的 验证 ， 既 包括 基于 服务 喜 
的 验证 ， 也 包括 使 用 PHP 的 验证 。 体 会 基本 的 HITP 验 证 和 目 己 设计 的 
验证 之 间 的 差别 。 


第 26 章 ”记录 并 监视 Web 服 务 硕 活动 


在 本 章 中 ， 你 将 学 到 : 


e 如 何 理解 Apache 日 专 格 式 和 上 日志 级别。 

。 如 何 备份 和 分 析 Apache 日 志 。 

o 加 何 解释 可 能 出 现在 日 志 中 的 第 见 钳 误 。 

。 如 何 创建 把 指定 项 目 记 录 到 数据 库 表 的 脚本 。 
。 如 何 创建 基于 这 些 日 志 表 的 目 定义 报告 。 


本 章 介 绍 Apache 中 的 日 志和 系统 如 何 工 作 ， 以 及 如 何 日 定义 日 志 系 
统 ， 包 括 要 存储 哪些 信息 以 及 信息 存储 在 哪里 。 男 外 ， 我 们 还 将 学 习 一 
种 快捷 方法 ， 以 使 用 PHP 和 和 MySQL 来 记录 Apache 日 志文 件 领 域 之 外 你 感 
兴趣 的 内 容 。 


26.1 标准 Apache 访 问 日 志 


使 用 Apache 的 基本 日 志 功 能 ， 我 们 可 以 通过 记录 对 答 主 Web 站 点 的 
服务 器 的 访问 ， 来 记录 谁 访问 了 Web 站 点 。 我 们 可 以 记录 浏览 器 请 求 和 
服务 器 相应 的 每 个 方面 ， 包 括 客 户 机 的 IP 地 址 、 用 户 以 及 所 访问 资源 。 
要 创建 一 个 请 求 日 志 ， 我 们 需要 如 下 3 个 步骤 。 


1. 定义 想 要 记录 什么 一 一 日 志 格 式 。 

2. 定义 将 其 记录 在 何 处 一 一 日 志文 件 、 一 个 数据 库 、 一 个 外 部 程 
FF e 

3. 定义 是 否 记 录 一 一 条 件 式 的 日 志 规 则 。 

下 和 面 几 节 进 一 步 讨论 这 些 步 又 。 
26.1.1 ”确定 记录 什么 


尽管 日 志 几 乎 涉及 到 和 请 求 相 关 的 每 个 方面 ， 我 们 可 以 通过 创建 一 
个 日 志 格式 来 定义 日 志 条 目 如 何 出 现 。 日 志 格式 就 是 一 个 字符 串 ， 其 中 
包含 了 和 日 志 格式 化 指令 混合 的 文本 。 日 志 格式 化 指令 以 a%6 打 头 ， 并 
且 后 面 跟着 一 个 指令 名 或 标识 符 ， 通 常 是 表示 要 记录 的 信息 片断 的 一 个 
字母. 


当 Apache 记 有 录 一 个 请 求 的 时 候 ， 它 扫 接 这 个 字符 串 并 且 用 值 蔡 代 每 
个 指令 。 例 如 ， 如 果 日 志 格 式 是 This is the client address %a, HEH 
束 是 类 似 This is the client address 10.0.0.2KJAZ. Ei, ARTs 


令 %a 由 发 出 请 求 的 客户 机 的 人 P 地 址 所 取代 。 表 26-1 提 供 了 所 有 格式 化 指 


令 的 一 个 完整 列表 。 
表 26-1 日 志 格 式 化 指令 


格式 化 选项 ii 明 


来 和 目 各 户 机 的 数据 


来 自 客户 机 的 远程 IP 地 址 
发 出 请 求 的 客户 机 的 主机 名 或 IP 地 址 。 是 否 记 录 主 机 名 取决 于 两 个 
KHR: 客户 机 的 IP 地 址 必须 通过 一 个 反 同 DNS 会 找 解 析 为 一 个 主机 
%h 名 ， 并 且 Apache 必 须 配 置 为 使 用 HostNameLookups 指 令 来 执行 此 奉 
R: 后 者 将 在 本 章 后 面 介绍 。 如 果 这 两 个 条 件 不 满足 ， 将 记录 客户 














机 的 IP 地 址 而 不 是 主机 名 
远程 用 户 ， 通 过 identd 协 议 获 取 。 这 个 选项 并 不 是 非常 有 用 ， 因 为 
在 大 多 数 客户 机 上 ， 并 不 支持 这 个 协议 
远程 用 户 ， 来 自 HTTP 基 本 验证 协议 
来 自 服 务 器 的 数据 
来 自 服务 器 的 本 地 Tp 地 二 


%D 服务 请 求 所 花 的 时 间 ， 以 微 秒 为 单位 
A 一 个 名 为 env_variable 的 环境 变量 的 值 (这 里 有 很 多 个 ) 


{env_variable}e 
















当前 时 间 。 如 果 {time_format} 存 在 ， 它 将 会 解释 为 UNIX strftime 到 
数 的 一 个 参数 。 参 见 Apache 手 册 的 logresolve 页 面 了 解 详细 内 容 


服务 请 求 所 花 的 时 间 ， 以 秒 为 单位 


应 答 请 求 的 服务 器 的 规范 的 名 称 
根据 UseCanonicalName 指 令 的 服务 右 名 称 


PARA at ERE ATR AS © TEVA ANI, EAA TE ARS ait A DA ACIS 
数据 之 前 ， 放 弃 连 接 。A+ 童 味 看 对 于 来 自 同 一 客户 机 的 进一步 请 


求 保 持 连 接 可 用 。A- 意 味 厦 连接 将 被 关闭 








来 目 请 求 的 数据 


O 
A e 一 个 名 为 cookie_name 的 cookie 的 值 
ŒE -ER 


请 求 协议 ， 如 HTTP 或 HTTPS 





% 来 自 客 户 机 的 请 求 中 ， 名 为 header name 的 一 个 标 头 的 全。 Ae 
{header_name}i | SAJBER H, PIG, Bidsevy lal A a EAS AY 4 AAR 


最 初 的 HTTP 请 求 的 文本 











%q 得 询 参数 ， 如 朱 有 的 话 ， 以 a ?为 前 绥 
请 求 的 URL， 没 有 得 询 参数 
用 于 HTTP 验证 《基本 的 或 摘要 的 ) 的 用 户 名 来 目 啊 应 的 数据 











发 回 给 客户 机 的 啊 应 的 主体 大 小 《不 包括 标 头 ) ， 以 字 节 为 单位 。 
、%B 两 个 选项 之 间 的 唯一 区 别 是 ， 如 果 没 有 数据 发 回 ，%b 将 记录 一 
个 -， 而 %B 将 记录 一 个 0 


服务 的 文件 的 路 径 ， 如 果 有 文件 的 话 


请 求 得 到 服务 的 时 间 


对 客户 机 的 响应 中 名 为 header_name 的 一 个 标 头 的 值 


人 


%>s 最 终 状态 代 但 。Apache 可 以 数 次 处 理 同 一 个 请 求 〈 内 部 重 定 同 ) 。 
这 是 最 终 啊 应 的 状态 代码 


通用 日 志 格 式 (Common Log Forma, CLF) a 日 志 格 
式 。 大 多 数 Web 站 点 可 以 使 用 这 一 格式 记录 请 求 ， 并 日， 这 一 格式 为 许 
多 日 志 处 理 和 报表 工具 所 理解 ， 其 格式 如 下 所 示 。 


‘sh %1 SU %t \"SP\" %>S Sb" 

也 就 是 说 ， 它 包括 客户 机 的 主机 名 或 IP 地 址 。 通 过 identd 的 远程 用 
户 ， 通 过 HTTP 验 证 的 远程 用 户 ， 请 求 得 到 服务 的 时 间 、 请 求 的 文本 、 
状态 代码 以 及 服务 的 内 容 的 字 节 大 小 。 











提示 : 








我 们 可 以 阅读 位 于 http:/www.w3.org/Daemon/User/Config/Logging.html 的 最 初 的 W3C 服 务 
器 的 通用 日 志 格 式 文档 。 





如 下 是 一 个 示例 的 CLF 条 目 。 


10.0.0.1 - - [19/Jan/2012:17:32:43 -@500] "GET / HTTP/1.0" 200 1101 


现在 我 们 准备 好 学 习 如 何 使 用 LogFormat 指 令 定 义 日 志 格 式 。 


指令 接受 两 个 参数 ， 第 一 个 参数 是 日 音字 符 串 ， 而 第 二 个 参数 是 将 要 和 
Ae PTT BIR PF 


例如 ， 来 日 默认 Apache 配 置 文件 的 如 下 指令 定义 了 CLF 并 且 为 其 分 


配 一 个 别名 common。 


LogFormat "sh 1 %u t \"Sr\" %>s b" common 


你 也 可 以 使 用 只 带 一 个 参数 的 LogFormat 指 令 ， 这 个 参数 要 么 是 一 
个 日 志 格 式 字 人 符 串 ， 要 么 是 一 个 别名 。 这 将 会 具有 为 TransferLog 指 令 所 
使 用 的 日 志 格 式 设 置 默 认 值 的 效果 ， 我 们 将 在 本 章 稍 后 介绍 这 一 点 。 


1. HostNameLookups 指 令 


当 一 个 客户 机 做 出 请 求 ，Apache 只 知道 客户 机 的 卫 地 址 。Apache 必 
须 执行 所 谓 的 反 癌 DNS 得 询 来 得 出 和 这 个 了 P 地 址 相关 的 主机 名 。 这 个 操 
作 可 能 项 费时 间 并 且 可 能 导致 请 求 处 理 的 显 圭 的 延 到 。 
HostNameLookups 指 令 人 允许 我 们 控制 是 否 执 行 反 同 DNS 查询。 


HostNameLookups 指 令 可 以 接受 如 下 参数 中 的 一 个 : on、off 或 
double， 默 认 值 是 of。double 得 询 参数 意味 着 Apache 将 通过 IP 找 到 主机 
名 ， 并 且 随 后 尝试 从 主机 名 找到 IP。 正 如 http://httpd.apache.org/docs- 
2.0/dns-caveats.html 所 介绍 的 ， 如 果 我 们 真 的 关心 安全 性 ， 这 个 过 程 是 
需要 的 。 如 果 我 们 要 使 用 主机 名 作为 Allow 和 Deny 规 则 的 一 部 分 ， 应 访 
执行 一 个 double DNS 查 询 而 不 管 HostrNameLookups 的 设置 。 


如 果 HostrNameLookups 是 打开 的 (on 或 者 double〉，Apache 将 会 记 
录 主 机 名 。 这 会 增加 服务 右上 的 额外 人 负担 ， 当 我 们 决定 打开 或 关闭 
HostNameLookups 的 时 候 ， 应 讼 意识 到 这 一 点 。 如 果 我 们 选择 保持 


HostNameLookups3</4], XX) F FF rey Vite AY i OR he ee Et BOA, 
Apache 将 只 记录 相关 的 IP 地 址 。 随 后 ， 有 众多 的 工具 可 以 用 来 解析 日 志 
EJIP. ta Ba pe) AS eA ye FB Apache ma” — so UA, SG SRO 
过 环境 变量 REMOTE_HOST 传 递 给 CGI 脚本 。 


2. IdentityCheck 指 令 


在 本 章 开始 处 ， 介 绍 了 如 何 通过 identd 协 议 使 用 %1 日 志 格式 化 指令 
来 记录 远程 用 户 名 。IdentityCheck 指 令 接 受 一 个 on 或 off 的 值 ， 来 打开 或 
关闭 对 这 个 值 的 检查 并 使 其 可 供 日 志 中 的 内 容 使 用 。 由 于 信息 并 不 可 靠 
并 且 要 伦 很 长 时 间 去 检查 ， 默 认 情 况 下 ， 这 个 选项 关闭 并 且 不 可 用 。 我 
们 提 到 %1， 只 是 因为 它 是 CLE 的 一 部 分 。 要 了 解 identd 协 议 的 更 多 内 
容 ， 可 以 参考 位 于 http://www.rfc-editor.org/rfc/rfc1413.txt 的 RFC 1413. 


3. 状态 但 


我 们 可 以 指定 是 个 和 在 日 将 条 目 中 记录 特定 的 元 系 。 在 本 草 开 始 ， 我 
们 学 习 了 以 a% 开 始 的 日 记 指 令 ， 其 后 跟 看 一 个 指令 标识 从 。 在 这 之 
站 ， 我 们 可 以 插入 一 个 状态 公 的 列表 ， 中 间 用 逗 写 隅 开 。 如 果 请 求 状态 
是 列 出 的 代码 中 的 一 个 ， 将 记录 访 参 效 ， 合 则 将 记录 -。 


例如 ， 如 下 的 指令 标识 符 将 记录 钳 误 请 求 〈 状 态 码 400) 以 及 市 有 
未 实现 的 方法 的 请 求 〈 状 态 码 501) 的 浏览 器 名 称 和 版 本 。 这 些 信息 对 
于 记录 哪个 客户 机 引 友 了 问题 很 有 用 。 
%400,501{User-agent}i 


我 们 可 以 在 方法 列表 的 前 面 加 上 一 个 “1*， 表 示 如 果 方 法 没有 实 
WM, MWK, 


*!409,501{User-agent}1 
26.1.2 ”记录 对 文件 的 访问 

记录 文件 是 Apache 中 记录 请 求 的 默认 方式 。 我 们 可 以 使 用 
TransferLog 和 CustomLog 指 令 来 定义 文件 名 。TransferLog 指 令 接受 一 个 
文件 参数 并 且 使 用 LogFormat 指 令 所 定义 的 最 独 的 日 志 格 式 ，LogFormat 
指令 只 帘 有 一 个 参数 别名 或 格式 字 从 串 ) 。 如 果 没 有 日 志 格 式 存 在 ， 
AUE ACLE. 


下 面 的 例子 展示 了 如 何 使 用 LogFormat 和 TransferLog 指 令 来 定义 一 
个 日 志 格式 ， 该 日 志 格式 以 CLE 为 基础 但 也 包含 浏览 器 名 称 。 


LogFormat "%h %1 %u %t \"%r\" >s %b \"%Sf{User-agent}i\"" 
TransferLog logs/access log 


CustomLog 指 令 人 允许 我 们 显 式 地 指定 日 六 格式 。 它 至 少 接受 两 个 参 
数 : 一 个 日 志 格 式 以 及 一 个 目标 文件 。 日 志 格 式 可 以 作为 别名 指定 ， 或 
者 直接 指定 为 一 个 日 志 字 人 符 串 。 

指令 示例 如 下 。 


LogFormat "h %l %u t \%r\" %>s %b \"%{User-agent}i\” myformat 
GustomLog Logs/access log myformat 


上 述 指令 和 如 下 指令 是 等 效 的 。 
CustomLog logs/access log "sh %1 %U %t \"%r\" %>s Sb \"%s{User-agent}i\"" 
CustomLog 指 令 接受 一 个 环境 变量 作为 第 三 个 参数 。 如 果 这 个 环境 


变量 存在 ， 条 目 将 记录 ， 人 否则 将 不 会 记录 。 如 果 环 境 变 量 通 过 一 
个 “> 前缀 求 反 ， 如 果 变 量 不 存在 ， 将 记录 条 目 。 


BAUS eas S Ota) ee E H SP cae GIF AIIPEG HL AR 


setEnvIf Request URI "{\.gif|\.jpgj}$" image 
GustomLog Llogs/access log common env=!image 


提示 : 





在 httpd.conf 文 件 的 此 处 和 其 他 区 域 中 用 于 模式 匹配 的 正则 表达 式 ， 和 PHP 以 及 其 他 编程 
语言 中 的 正则 表达 陈 放 从 同样 的 格式 。 


26.1.3 ”记录 对 一 个 程序 的 访问 


TransferLog 和 CustomLog 指 令 都 可 以 接受 一 个 以 管道 符 | 为 前 绥 的 可 
执行 程序 作为 一 个 参数 。Apache 将 会 把 日 志 条 目 写 入 程序 的 标准 输入 。 
反 过 来 ， 这 个 程序 将 会 处 理 输 入 ， 例 如 把 条 目 记 录 到 数据 库 ， 或 将 它们 
FEI BI AP ARG 


如 果 访 程序 由 于 某 些 原 因而 停止 ， 服 务 右 将 确 你 它 香 新 启动 。 如 来 
服务 喜 死机 ， 程 序 也 会 仓 止 。 绑 定 到 Apache 的 rotatelogs 工 具 ， 驳 是 日 志 
程序 的 一 个 例子 ， 我 们 将 在 本 草 后 面 介 绍 它 。 作 为 一 个 通用 的 规则 ， 除 
非 有 了 使 用 一 个 特定 程序 的 具体 需求 ， 含 则 很 容易 并 且 更 可 能 记录 到 磁 
盘 上 的 一 个 文件 ， 并 且 随 后 可 能 在 另外 一 从 不 同 的 机 霹 上 进行 处 理 、 合 


并 、 日 忘 分 析 等 。 


确保 用 来 记录 请 求 的 程序 古 安 全 的 ， 因 为 它 作 为 Apache 所 局 动 的 一 
个 用 户 运 行 。 在 UNIX 上 ， 这 通常 意味 着 root， 因 为 外 部 程序 在 服务 妖 改 
杰 将 其 用 户 ID 改 为 User 指 令 的 值 之 前 局 动 ， 通 音 User 指 令 的 值 为 nobody 


或 www。 


26.2 ”标准 Apache 错 误 日 志 


除了 记录 客户 请 求 ，Apache 还 可 以 配置 为 记录 错误 信息 以 及 调试 信 
。 除 了 Apache 目 有 身 产 生 的 错误 ，CGI 错 误 也 可 以 记录 。 


每 个 错误 日 志 条 目前 面 都 会 有 错误 发 生 的 时 间 以 及 客户 机 的 IP 地 址 
或 主机 名 ， 如 果 有 这 些 信息 的 话 。 和 HTTP 请 求 记录 一 样 ， 我 们 可 以 把 
错误 信息 记录 到 一 个 文件 或 程序 中 。 在 UNIX 系 统 上 ， 我 们 也 可 以 记录 
到 syslog 守 护 进 程 。 在 Windows 上 ， 错 误 可 以 记录 到 Windows 事 件 日 忘 
并 且 可 以 通过 Windows 事 件 合 看 需 租 看 。 使 用 ErrorLog 指 令 来 定义 要 记 
录 到 哪里 。 


26.2.1 ”把 错误 记录 到 一 个 文件 


一 个 文件 参数 给 出 了 铺 误 日 志文 件 的 路 径 。 如 末 这 个 路 径 是 相对 
的 ， 假 设 它 相对 于 服务 器 根 上 目录。 默认 情况 下 ， 错 误 日 志文 件 位 于 logs 
目录 下 ， 并 且 在 UNIX 上 名 为 error_ log， 在 Windows 上 名 为 error.log。 下 
面 是 一 个 例子 。 


ErrorLog Llogs/my error log 
26.2.2 ”把 错误 记录 到 一 个 程序 


我 们 可 以 指定 到 一 个 程序 的 路 径 ， 用 管道 符 沾 "作为 前 缀 。Apache 将 
会 把 错误 记录 到 该 程序 的 标准 输入 中 ， 并 且 ， 访 程序 会 进一步 处 理 它 
们 。 下 和 面 是 一 个 例子 。 


ErrorLog "|/usr/local/bin/someprogram" 


26.2.3 syslog T I HEA% 


在 UNIX 系 统 上 ， 如 有 果 我 们 指定 了 syslog 作 为 一 个 参数 ， 可 以 把 错误 
消息 记录 到 UNIX 系 统 日 总 守护 进程 syslogd。 默 认 情 况 下 ， 日 志 钳 误 记 
录 到 syslog 工 具 local7。 这 个 工具 是 系统 产生 的 错误 的 一 部 分 。 我 们 可 以 
通过 提供 syslog:facility 作 为 一 个 参数 来 指定 一 个 工具 。syslog 工 具 的 例 
子 有 mail、uucp、local0、locall 等 。 要 获取 完整 的 列表 ， 请 奏 看 系统 所 
包含 的 syslog 的 文档 (在 命令 行 尝 试 man syslogd 或 man syslogd.conf) 。 
下 面 是 记录 到 syslog 的 一 个 例子 。 


ErrorLog syslog: Local6 


26.2.4 LogLevel 指 令 


Apache 所 提供 的 错误 信息 有 几 种 重要 程度 ， 我 们 可 以 选择 只 记录 至 
要 的 信息 ， 而 急 略 提示 信息 或 无 天 案 要 的 警告 信息 。LogLevel 指 令 接 受 
一 个 错误 级 唱 参 数 。 只 有 重要 级 别 或 更 高 级 别 的 错误 才 会 记录 。 

表 26-2 为 LogLevel 指 令 指 定 了 有 效 仁 ， 束 像 Apache 文 档 中 所 指定 的 
一 样 。 默 认 情 况 下 ，LogLevel 的 值 是 warn。 对 于 大 多 数 Apache 安 装 来 
说 ， 这 应 该 足够 了 。 如 果 你 要 尝试 对 一 个 特定 配置 进行 故障 排除 ， 可 以 
把 级 别 修 改 为 debug。 


表 26-2 Apache sch Fh MN LogLeveliz Ii 


Emergencies (紧急 ) 
系统 不 可 用 A 


Child cannot open lock file. Exiting 





alert | 必须 立即 采取 措施 getpwuid: couldn’t determine user name from uid 


Crit 重要 条 件 socket: Failed to get a socket, exiting child 


error | 错误 条 件 Premature end of script headers 


Child process 1234 did not exit, sending another 


notice | 一 般 但 重要 条 件 httpd: caught SIGBUS, attempting to dump core ina 


Server seems busy, (You may need to increase 


StartServers, or Min/MaxSpareServers) 


debug | JH WAFS Opening config file... 





26.3 ”管理 Apache 日 志 


Apache 提 供 了 几 种 工具 来 管理 日 志 。 其 他 的 Apache 专 用 的 第 三 方 工 
上 其 也 可 以 使 用 ， 并 有 昌 这 里 也 提 到 了。 由 于 Apache 可 以 以 CLF 记 了 杂 请 求 ， 
大 部 分 通用 日 总 处 理工 具 也 可 以 用 于 Apache。 


26.3.1 解析 主机 名 


在 本 章 前 面 ， 我 们 学 习 了 如 何 使 用 HostNameLookups 指 令 在 做 出 请 
求 的 时 候 打 开 或 天 财主 机 名 解析 。 如 采 HostNameLookups 议 置 为 o 呈 〈 默 
认 值 )， 日 志文 件 将 只 包 仿 IP 地址。 随后 ， 我 们 可 以 在 UNIX 上 使 用 命 
令 行 logresolve 工 具 或 者 在 Windows 上 使 用 logresolve.exe 来 处 理 日 志文 
件 ， 并 且 把 IP 地 址 转换 为 主机 名 。 


logresolve 工 其 从 标准 输入 读 取 日 志 条 目 ， 并 且 把 结果 输出 到 其 标准 
输出 。 要 读 取 一 个 文件 或 写 入 到 一 个 文件 ， 我 们 在 UNIX 和 Windows 上 
者 可 以 使 用 重 定 同 ， 如 下 所 示 。 


logresolve < access, Logd > resolved. log 


日 志 解 析 工 具 效率 很 高 ， 因 为 它们 在 响应 客户 机 请 求 的 时 候 可 以 组 
存 结果 并 且 不 会 引起 任何 延迟 。 


26.3.2 日 志 备 份 


在 高 流量 的 Web 站 氮 中 ， 记 录 访 问 的 日 忘 文件 可 能 很 快 融 变 得 很 
大 。 我 们 应 该 有 一 种 方法 来 定期 备份 日 过， 按照 定义 的 时 间 间 隔 来 保存 
FF GAY H BAE o 


fEApacheia{T WIN, HELA, AAR as Bes 
入 到 它们 。 解 决 方案 是 使 用 一 个 中 间 程 序 来 记录 请 求 。 反 过 来 ， 这 个 程 
序 将 负责 日 总 的 备份 。 


Apache 提 供 了 UNIX 上 的 rotatelogs 程 序 和 Windows 上 的 rotatelogs.exe 
程序 来 完成 此 任务 。 它 接受 3 个 参数 : 一 个 文件 名 ， 一 个 以 秒 为 单位 的 
备份 周期 ， 以 及 一 个 可 选 鸭 、 相 对 UTC(Coordinated Universal Time， 协 
调 世 界 时 ) 的 分 钟 仿 移 量 ， 示 例如 下 。 

TransferLog "|bin/rotatelogs /var/logs/apachelog 86400" 

每 天 都 创建 了 一 个 狐 的 日 志文 件 ， 并 且 把 当前 日 志 移 动 

全 /var/logs 《在 命令 的 末尾 ，86400 束 是 一 天 中 的 秒 数 ) 。 


提示 : 


如 条 程序 的 路 径 中 包含 空格 ， 我 们 可 能 需要 通过 在 它们 前 面 加 上 一 个 \ 《友和 斜 枉 ) 来 转 
义 ， 例 如 ，MyYNDocuments。 这 在 Windows 平 台 上 尤其 常见 。 


如 末 文 件 名 包含 % 前 组 选项 ， 名 字 将 作为 strftime 国 数 的 输入 处 理 ， 
该 函数 把 % 选 项 转换 为 时 间 值 。rotatelogs 工 具 的 手册 页 面包 含 了 一 个 完 
整 的 选项 列表 ， 下 面 是 一 个 例子 。 


TransferLog "|bin/rotatelogs /var/logs/apachelog&sm %d %y 86400" 
这 个 命令 把 当前 月 份 、 日 期 和 年 份 添 加 到 日 志文 件 名 中 。 


如 朵 这 个 名 字 设 有 包含 任何 % 格 式 的 选项 ， 以 秒 为 单位 的 当前 时 间 
将 会 深 加 到 备份 文件 的 名 字 中 。 


26.3.3 日 志 分 析 


IM eA ATS RA te SOC, BEA PRS a EE 
MA EH, RIIKE SAE, Ma Wor EMI RRA Riise 
和 访问 者 行为 的 信息 。 

有 很 多 商业 软件 、 共 至 软件 以 及 目 由 软件 应 用 程序 可 以 用 来 进行 日 
志 分 析 和 报告 。 两 浆 最 流行 的 开源 应 用 程序 是 
Webalizer(http://www.mrunix.net/webalizer/ ) 和 awstats (http://awstats. 


sourceforge.net/ )。Wusage 是 一 于 物美 价 廉 的 商业 软件 ， 可 以 
在 http://www.boutell.com/wusage/ 找到 。 


26.3.4 ”监视 错误 日 志 
如 果 我 们 在 UNIX 系 统 上 运行 Apache， 可 以 使 用 tail 命 令 行 工 具 来 监 
视 实 时 的 日 志和 条目， 不 管 是 访问 日 志 还 是 错误 日 志 都 可 以 ， 语 法 如 下 。 


tail -f logname 


HHF, lognameze Apache Fas CFA ERIE. ERKEL AN H es 
MEN Be ILT, FFA 4A A SBC PEAY (RARE SN EI. 


我 们 可 以 找到 其 他 的 程序 ， 使 我 们 通过 扫 摘 错误 日 志文 件 ， 找 到 指 
定 错 误 、 错 误 请 求 等 ， 来 识别 问题 并 报告 它们 。ScanErrLog 束 是 这 样 的 


一 个 程序 ， 可 以 在 http://www.librelogiciel. com/software/ 找到 它 。 


26.4 把 目 定 义 信息 记录 到 一 个 数据 库 


在 MySQL 中 创建 自己 的 日 志 表 ， 配 合 PHP 代 码 片断 ， 就 可 以 帮助 我 
们 捕获 对 站 点 的 具体 页 面 的 访问 相关 的 信息 。 使 用 这 些 信息 ， 我 们 可 以 
创建 自 定义 报表 。 这 种 方法 比 困难 读 取 Apache 日 志文 件 更 简单 一 些 ， 万 
其 是 当 我 们 要 查找 访问 信息 的 一 个 子 集 的 时 候 。 下 面 各 节 介绍 了 这 一 过 
程 的 一 个 简单 版 本 。 


26.4.1 创建 数据 库 表 


目 定义 日 总 方法 的 第 一 步 是 创建 数据 库 表 。 如 下 的 表 创 建 命 令 在 
MySQL 数 据 库 中 创建 了 一 个 名 为 access_tracker 的 表 ， 其 中 有 ID、 页 面 标 
题 、 用 户 代 理 和 访问 日 期 的 字段 。 

CREATE TABLE access_tracker ( 
id INT NOT NULL PRIMARY KEY AUTO INCREMENT, 
page title VARCHAR (50), 
user_agent TEXT, 


date _ accessed DATE 


); 
fe PORK, RITE VAGUE XPS HES AWN AAG EZ. 
26.4.2 ”创建 PHP 代 人 三 段 


按照 我 们 已 有 经 验 ， 代 码 段 基本 上 只 需要 一 小 段 代 码 。 换 名 话说， 
代码 段 只 执行 简单 的 任务 ， 因 而 并 不 是 一 个 长 脚本 。 在 这 个 例子 中 ， 程 
序 清单 26.1 中 的 代码 段 把 一 些 基 本 信息 写 入 到 access_tracker 表 。 


程序 清单 26.1 访问 记录 的 代码 段 


<?php 

{iset up static variables 

$page title = "sample page A"; 

$user agent = getenv( HTTP_USER_AGENT’); 


iconnect to server and select database 
$mysqli = mysqli connect{"Llocalhost", “joeuser", "“Somepass", "testDB") 
or die(mysql_error()); 


oOn~ aM oa Fw he — 


9: 

10: //create and issue query 

11: $sql = "INSERT INTO access tracker (page title,user agent,date accessed) 
12: VALUES ('Spage title', 'Suser_agent', now{))"; 

13: $result = mysqli query($mysqli, $sql) or die({mysqli error($mysqli)}; 

14: 

15: /Close connection to MyS@L 

16: mysqli _close($mysq1i); 

17: %> 


这 段 代码 的 用 法 很 简单 : 将 它 放 置 在 想 要 记录 的 每 个 页 面 的 开始 
处 。 对 于 每 个 页 面 ， 把 代码 段 中 的 $page_tite 的 人 修改 为 页 和 面 的 实际 标 


E o 


现在 ， 创 建 名 为 sample1.php 的 一 个 示例 脚本 ， 其 中 包 全 了 程序 清单 
26.1 的 内 容 ， 如 下 是 程序 清单 26.2 的 其 他 内 容 。 


程序 清单 26.2 ”示例 HTML 页 面 


<!DOCTYPE html> 

<html> 

<head> 

<title>Sample Page A</title> 
</head> 

<body> 

<h1i>Sample Page A</h1> 
<p>blah blah blah.</p> 
</body> 

ð: </html> 


创建 这 个 文件 的 数 个 副本 ， 使 用 不 同 的 文件 名 和 不 同 的 $page_title 
值 。 然 后 ， 使 用 Web 济 虹 剖 访问 这 些 不 同 的 足 面 ， 以 填 元 日 志 表 。 


— O ON OOF Wh 一 


26.4.3 ”创建 示例 报表 


当 access_tracker 表 拥有 了 数据 ， 可 以 创建 一 个 简单 的 报表 页 面 来 发 
布 这 些 信 息 。 程 序 清单 26.3 中 的 代码 创建 了 一 个 报表 ， 它 执行 查询 来 计 
算 结 果 以 及 使 用 的 浏览 器 的 概要 。 这 些 代 码 段 将 在 程序 清单 之 后 说 明 。 


程序 清单 26.3 ”创建 一 个 访问 报表 


ons) Dt Of wh — 


<?php 

//connect to server and select database 

amysqli = mysqli connect("localhost", “‘joeuser", Somepass ， "“testDB"} 
or die(mysqli error{)); 


//1ssue query and select results for counts 

$count_sql = "SELECT count(page_ title) AS p count FROM access tracker"; 

$count res = mysqli query(Smysqli, $count sql) or 
die(mysqli_error($mysqli}}; 


while ($count info = mysgli fetch array($count res)) { 
$all_count = $count_info['p_count']; 


} 


//issue query and select results for user agents 

suser_agent_sql = ‘SELECT DISTINCT user_agent, count(user_agent} AS 
Ua count FROM access tracker GROUP BY user agent 
ORDER BY uUa_count desc"; 

$user agent res = mysqli gquery($mysqli, $user agent sql) 
or die(mysgli_ error($mysqli)); 


//fstart user agent display block 
$user agent block = "<ul>"; 


:iloop through user agent results 
while ($row ua = mysqli fetch array($user agent res)) { 
$user agent = $row ua[ ‘user_agent’ ]; 
$user agent count = $row ua[ ua count ]; 
$user agent block .= ' 
<li>" .$user_agent. " 
<ul> 
<li><em>accesses per browser: ".$user_agent_count. </em> 
</Ul> 
</li>’; 


} 


/ffinish up the user agent block 
$user_agent_block 了 一 "</ul>"; 


//1ssue query and select results for pages 
$page title sql = ‘SELECT DISTINCT page title, count(page title} AS 


41; pt_count FROM access tracker GROUP BY page title 


42: ORDER BY pt count desc"; 

43: $page title res = mysqli query({$mysqli, $page title sql) 
44: or die{mysqli_error($mysq1li} ); 

45: 


46: //start page title display block 

47: $page title block = "<ul>"; 

48: 

49: {/loop through results 

5@: while ($row pt = mysqli _fetch_array($page title res)) { 


51; $page title = $row pt['page title']; 
52: $page_count = $row pt[ pt count ]; 
aoe $page title block .= '" 

54: <li>". $page title." 

55. <ul> 

56: <li><em>accesses per page: ".$page count. '"</em> 
57: </ul> 

58; < TI | 

59: } 

60: 

61: //finish up the page title block 

62: $page title block .= "</ul>"; 

63: 


64: //close connection to MyS@L 
65: mysqli claose($mysqli); 


66: 7> 

67: <!DOCTYPE html> 
68: <html> 

69: <head> 


70i <title>Access Report</title> 

7i: </head> 

f2: <body> 

#3: <hi>Access Report</h1> 

74: <p><strong>Total Accesses Tracked: </strong= 
75: <?php echo "$all count"; ?></p> 

76: <p><strong>Web Browsers Used:</strong> 
77: <?php echo "$user agent block"; ?></p> 
78: <p><strong>Individual Pages:</strong> 
79: <?php echo "$page title block"; ?></p> 
8@: «</body> 

81: </html> 


第 3 行 连接 到 数据 库 ， 以 便 我 们 可 以 根据 access_tracker 表 来 执行 得 
询 。 第 7 行 到 第 8 行 执行 查询 来 统计 页 面 的 总 数 ， 第 15 行 到 第 19 行 计算 用 
户 代 理 访问 的 次 数 。 第 22 行 开始 一 个 无 序列 表 语 句 块 ， 用 于 用 户 代 理 查 
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行 结 


第 40 行 到 第 44 行 创建 并 执行 一 个 查询 来 统计 单独 的 页 面 。 第 47 行 开 
始 一 个 无 序列 表 语 句 块 ， 用 于 显示 这 一 查询 的 结果 ， 第 50 行 到 第 59 行 裔 
历 这 个 结果 并 创建 访问 过 页 面 的 列表 ， 这 个 无 序列 表 块 在 第 62 行 结 


把 这 些 代 码 放 入 到 一 个 名 为 accessreport.php 的 文本 文件 中 ， 然 后 将 
其 放 入 到 Web 服 务 器 文档 根 目 录 下 。 当 访问 这 个 报表 时 ， 将 会 看 到 如 图 
26-1 所 示 的 结果 ， 页 面 名 称 、 计 数 和 浏 响 占 信 息 可 能 有 所 不 同 ， 但 情况 
大 臻 是 这 样 。 


Access Report 
= © |© http://localhost/26/accessreport.php 


Access Report 


Total Accesses Tracked: 40 
Web Browsers Used: 


e Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit'535.7 (KHTML, like Gecko) 
Chrome'16.0.912.75 Safari $35.7 
© accesses per browser: 19 
* Mozila/5.0 (Windows NT 6.1; WOW64; rv-9.0.1) Gecko/20100101 Firefox/9.0.1 
© accesses per browser: 13 
e Mozilla’5.0 (compatible; MSIE 9.0; Windows NT 6.1; WOW64; Trident’5.0) 
© accesses per browser: 6 


Individual Pages: 


* sample page B 


o accesses per page: 25 


è sample page A 


© accesses per page: I} 





图 26-1 记录 页 面 的 目 定义 访问 报表 


这 种 记录 比 羊 震 谈 取 Apache 访 问 日 志 要 容易 很 多 ， 但 是 我 们 不 建议 
使 用 数据 库 驱 动 的 系统 来 完全 蔡 代 访问 日 志 。 即 便 MySQL 在 系统 上 能 
够 工作 得 很 好 ， 数 据 库 连 接 也 需要 很 大 的 开销 。 相 反 ， 把 你 的 页面 记录 
定 为 目标 则 特别 重要 。 


26.5 ”小 结 


本 章 介 绍 了 如 何 记录 Apache 所 产生 的 有 关 请 求 和 错误 的 特定 信息 。 
我 们 可 以 把 日 志 存 储 到 文件 或 数据 库 中 ， 或 者 将 它们 传递 给 外 部 程序 。 
我 们 笠 习 了 可 以 用 来 管理 、 处 理 和 分 析 日 总 的 不 同 工 具 ， 既 有 那些 包含 
在 Apache 中 的 工具 ， 也 有 那些 来 目 第 三 方 的 可 用 工具 。 


最 后 ， 我 们 看 到 了 一 种 简单 的 方法 ， 它 使 用 PHP 代 码 段 和 一 个 
MySQL 数 据 库 来 执行 特定 页 面 的 简单 访问 记录 。 这 些 信息 随后 以 一 个 
用 PHP 生 成 的 、 简 单 的 访问 报表 的 形式 最 示 出 来 。 


26.6 Q&A 


Q: 为 什么 不 布 望 记录 图 像 ? 


A: 在 负载 较 重 的 服务 器 上 ， 日 志 可 能 成 为 一 个 瓶颈 。 如 果 日 志 的 
目的 是 为 了 记录 访客 的 数量 以 及 分 析 他 们 对 网 站 的 使 用 ， 我 们 可 以 通过 
只 记录 HTML 页 面 来 得 到 结果 ， 而 不 用 把 图 像 包含 在 其 中 。 这 会 减少 存 
储 在 日 志 中 的 信息 的 数目 ， 并 且 减 少将 其 写 入 日 志 的 时 间 。 


26.7 ”实践 练习 


这 个 实践 练习 设计 用 来 帮助 你 预测 可 能 的 问题 、 复 习 已 经 学 过 的 知 
识 ， 并 且 开 始 把 知识 用 于 实践 。 


1. BU fy aE a ME E A RVT Td PRT Web ait K HI 2 7 i EY fis SA 
记 入 日 志 ? 


2. 如 何 把 图 像 记 录 为 一 个 不 同 的 文件 ? 


3. 为 什么 硕 望 在 你 的 Apache 配 置 中 关闭 HostNameLookupsoff? 


fi E 


1. ÆRE F, RITE REN ELER H RNR E NEK, 
例如 目 己 的 请 求 ， 以 便 它 们 不 会 改变 结 东 。 我 们 可 以 通过 后 处 理 日 专 删 
除 它们 ， 或 者 使 用 SetEnvIf 指 令 来 做 到 ， 如 下 所 示 。 


oy 


SetEnvIf directive: 


SetEnvIf Remote Addr 10\.@\.@\. intranet 
CustomLog logs/access log ‘Ssh sl %u st \%r\" %>s %b lintranet 


2. 在 本 章 前 面 ， 我 们 学 习 了 如 何 避 免 记 录 图 像 。 采 用 同样 的 环境 
变量 的 方法 ， 我 们 可 以 把 图 像 记 录 到 另 一 个 文件 ， 而 不 是 完全 忽略 图 
像 ， 示 例如 下 。 


SetEnvIf Request URI "{\.gif|\.jpeg})$" image 
CustomLog logs/access log common env=!image 
CustomLog logs/images log common env=image 


3. 打开 HostNameLookupson 会 导致 服务 器 上 额外 的 负担 ， 因 为 它 
查找 访问 站 点 的 用 户 的 IP 并 将 其 写 入 到 日 志文 件 。 稍 后 的 日 子 ， 在 产生 
使 用 报告 的 时 候 ， 你 仍然 可 以 使 用 主机 名 解析 程序 来 获得 信息 ， 并 由 此 
将 针对 用 户 的 服务 器 负载 最 小 化 。 


EA wel 


1. 创建 一 个 记录 脚本 ， 它 记录 了 通过 PHP 对 一 个 数据 库 的 访问 。 
使 用 可 能 的 环境 变量 的 列表 ， 记 录 页 面 标题 、 用 户 代 理 和 访问 日 期 以 外 
更 多 的 信息 。 


2. 创建 你 所 和 存储 的 访问 数据 的 一 个 报表 ， 湛 加 一 个 日 期 艺 围 选择 
aes FP AAA Bb PE AE ZA 


第 27 章 ”应 用 程序 本 地 化 


在 本 章 中 ， 我 们 将 学 习 : 


。 如 何 识 列 不 同 的 字符 集 并 为 此 做 好 准备 。 
。 如 何 准 备 应 用 程序 的 结构 并 生成 本 地 化 站 后。 


World Wide Web 中 的 关键 词 束 是 World Wide CRATE FG) 。 使 
用 PHP 和 MySQL 创 建 一 个 可 供 讲 不 同 语言 的 人 使 用 的 Web 站 点 ， 这 是 小 
沪 一 供 。 人 准备 让 你 的 应 用 程序 在 多 种 地 方 使 用 的 过 程 ， 叫 做 国际 化 ; 为 
每 一 个 地 方 而 目 定义 代码 的 过 程 叫做 本 地 化 。 


27.1 天 于 国际 化 和 本 地 化 


首先 ， 不 管 是 国际 化 还 是 本 地 化 ， 都 是 与 翻译 类 似 的 同一 件 事 情 。 
实际 上 ， 我 们 可 以 完全 地 翻译 一 个 web 站 点 ， 全 部 用 德语 、 全 部 用 日 
语 ， 或 者 全 部 用 我 们 想 要 的 任何 一 种 语言 ， 并 且 它 不 是 考虑 为 一 个 国际 
化 或 本 地 化 的 Web 站 点 。 它 只 是 一 个 翻译 后 的 站 点 。 国 际 化 的 关键 部 分 
如 下 所 示 。 


。 使 所有 的 字符 串 、 图 标 和 图 形 外 部 化 。 
。 修改 格式 化 函数 〈 日 期 、 货 币 、 数 字 等 ) 的 显示 。 


我 们 构建 了 目 己 的 应 用 程序 以 使 得 字符 串 外 部 化 〈“ 即 函数 、 类 和 其 
他 脚本 中 使 用 的 所 有 字符 串 在 一 个 地 方 管理 ， 并 且 作 为 常量 来 导入 或 引 
用 ) ， 并 且 格 式 化 函数 可 以 随 着 本 地 设置 而 变化 之 后 ， 就 开始 了 本 地 化 


的 过 程 。 翻 详 正 古 这 个 过 程 的 一 部 分 。 


一 个 本 地 设置 实际 上 是 一 个 组 ， 在 这 个 例子 中 ， 是 一 组 翻译 后 的 字 
从 串 、 图 形 、 文 本 以 及 格式 化 惯例 ， 它 们 将 用 于 那些 要 本 地 化 的 应 用 程 
厅 或 Web 站 点 。 这 些 组 通常 是 用 应 用 程序 所 使 用 的 语言 的 名 字 米 引用 ， 
例如 German locale。 尽 管 很 明显 German locale 包 含 了 翻译 为 德语 的 文 
本 ,但 并 不 意味 看 这 个 站 点 只 能 供 德 国人 使 用 ， 说 德语 的 奥地利 人 也 可 
以 使 用 一 个 本 地 化 的 德语 站 点 ， 但 是 ， 并 没有 称 其 为 Austrian locale. 


在 下 面 的 几 市 中 ， 我 们 学 习 了 如 何 使 用 不 同 字 符 集 以 及 如 何 修 改 环 
境 ， 以 成 功 地 做 好 应 用 程序 本 地 化 的 准备 。 


27.2 ”关于 字符 集 


子 付 集 通 归根 据 一 种 语言 中 用 来 定义 一 个 字符 所 需 的 学 市 的 数目 ， 
分 为 单字 节 字 符 集 和 多 字 节 字符 集 。 英 语 、 德 语 和 法 语 等 还 有 很 多 其 他 
的 语言 都 是 单字 节 语 言 ， 表 示 像 字母 a 或 数字 9 这 样 的 一 个 字符 ， 只 需要 
一 个 字 节 。 单 字 节 代码 集 最 多 有 256 个 字符 ， 包 括 ASCII 字 符 的 完整 集 
合 、 重 音字 符 和 其 他 所 需 的 格式 字符 。 


多 字 市 代码 集 拥 有 多 于 256 个 的 了 字符， 包含 所 有 的 早 子 市 学 从 作为 
其 子 集 。 多 字 节 语言 包括 繁体 中 文 和 简体 中 文 、 日 文 、 韩 文 、 泰 文 、 阿 
拉 伯 语 、 和 硕 伯 来 语 等 等 ， 这 些 语言 者 需要 多 于 一 个 字 节 来 表示 一 个 字 
符 。 一 个 好 的 例子 就 是 单词 Tokyo， 即 日 本 的 首都 东京 。 在 英语 中 ， 它 
拼写 为 5 个 不 同 的 字母 ， 一共 使 用 5 个 字 市 。 然 而 ， 在 日 语 中 ， 这 个 早 词 
用 两 个 音节 表示 ， 分 别 是 tou 和 kyou， 每 个 音节 使 用 两 个 字 节 ， 一 共用 


到 4 个 字 节 。 


这 怠 是 字符 集 及 其 背后 的 技术 的 一 个 简要 介绍 ， 但 是 ， 实 用 性 在 
于 : 为 了 以 原本 的 语言 正确 地 解释 和 显示 Web 页 面 中 的 文本 ， 告 诉 Web 
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集 的 正确 标 涉 ， 在 那些 主语 言 不 是 日 语 的 Web 浏 览 器 中 将 错误 地 显示 这 
些 页 面 。 换 句 话说 ， 由 于 没有 包含 字符 集 人 信息， 浏览 器 假设 用 自己 的 默 
认 字 人 符 集 来 显示 文本 。 例 如 ， 如 末日 语 页 面 使 用 Shift JIS 或 UTF-8 字 符 


集 ， 并 且 浏 览 器 设置 为 TSO-8859-1， 你 的 浏览 器 将 会 尝试 使 用 单字 节 

ISO-8859-1 字 符 集 来 显示 日 语文 本 。 这 将 会 非常 糟糕 ， 除 非 标 头 提 醒 它 
使 用 Shift_JIS 或 UTF-8， 并 且 你 的 操作 系统 上 安装 了 相应 的 库 和 语言 

包 。 


Mojibake 是 表示 这 种 不 能 识别 的 字符 的 一 个 术语 。 要 了 解 更 多 信 
已， 参见 http:Wen.wikipedia.org/wiki/Mojibake 。 所 涉及 的 标 头 就 是 
Content-type 和 Content-language 标 头 ， 它 们 也 设置 为 META 标 记 。 由 于 我 
们 拥有 了 一 个 动态 环境 的 所 有 工具 ， 了 最 好 在 文本 之 前 肥大 相应 的 标 头 ， 
并 且 在 文档 中 显示 正确 的 META 标 记 。 下 面 是 header0 函 数 的 一 个 例子 ， 
它 输 出 了 一 个 英文 站 点 的 正确 的 字符 信息 。 


headert Content-Type: text/html;charset=IS0-8859-1"}; 
headert 'Content-Language: en"); 


相应 的 HTML5 标 签 如 下 所 示 。 


«xhtml lang="en'> 
<meta charset='IS0-8859-1'"> 


一 个 德语 站点 将 会 使 用 相同 的 字符 集 ， 但 语言 代码 不 同 。 


headert Content-Type: text/html;charset=IS0-8859-1"}; 
headert 'Content-Language: de"); 


相应 的 HTML5 标 签 如 下 所 示 。 


«html lang="de > 
<meta charset= IS0-8859-1°> 
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header{"Content-Type: text/html;charset=Shift JIS }; 
header("Content-Language: ja"); 


相应 的 HTML5 标 签 如 下 所 示 。 


<html lang="ja"> 
<meta charset="Shift_JIS"> 


27.3 环境 修改 


在 本 书 的 安 少 章节 中 所 定义 的 环境 ， 需 要 进行 修改 以 处 理 本 地 化 站 
点 。 尽 管 我 们 可 以 在 Apache、PHP 和 MySQL 中 使 用 多 个 和 语言 相关 的 设 
置 ， 以 满足 本 地 化 站 点 的 需求 ， 但 我 们 也 可 以 不 对 配置 做 任何 和 语言 相 
天 的 改变 ， 从 而 执行 本 章 中 的 所 有 任务 。 针 对 你 自己 的 信息 ， 下 面 各 市 
将 介绍 为 国际 化 而 使 用 Apache、PHP 和 MySQL 的 相应 文档 。 


27.3.1 ”Apache 的 配置 修改 


在 第 29 半 ， 我 们 将 学 习 使 用 mod_mime 或 mod_negotiation 模 块 以 及 
AddLanguage 和 AddCharset 指 令 〈 还 有 其 他 指令 ) 来 进行 内 容 协商 的 概 
念 。 当 我 们 手动 改变 了 文件 的 扩展 名 ， 并 且 和 希望 Apache 根 据 这 一 扩展 名 
来 解释 要 使 用 的 字符 集 的 时 候 ， 需 要 用 到 这 些 指令 。 然 而 ， 这 并 非 本 重 
讨论 的 话题 。 我 们 和 希望 你 的 所 有 本 地 化 站 点 都 具有 相同 的 文件 命名 惯例 

(例如 ，index.html 和 company_info.html) ， 而 不 是 必须 手动 地 创建 具有 
基于 不 同 语言 的 扩展 名 的 多 个 页 面 来 提供 翻 详 的 文件 。 我 们 对 于 站 点 本 
地 化 的 目标 是 ， 在 一 个 Web 服务 左上 有 一 组 使 用 正确 翻译 的 文本 填充 的 
页 耐 在 运行 。 


提示 : 








基于 Apache 的 内 容 协 商 使 用 共有 基于 语言 的 命名 惯例 的 多 个 文件 并 没有 错 。 只 不 过 这 不 
是 本 章 关 注 的 重点 。 我 们 可 以 在 http:Whttpd.apache.org/ docs.2.0/content.negotiation.html 阅读 到 
有 关 基 于 Apache 的 内 容 协商 的 更 多 内 容 。 


27.3.2 PHP 的 配置 修改 


和 Apache 一 样 ， 对 于 本 草 的 任何 任务 ， 也 不 需要 在 PHP 中 做 出 配置 
改变 。 然 而 ， 如 果 需 要 的 话 ， 我 们 可 以 使 用 很 多 和 处 理 多 字 市 字符 相关 
的 六 数 。 这 些 函 数 可 以 在 位 于 http://www.php.net/mbstring 的 PHP 手 册 中 
找到 ， 并 且 必 须 在 配置 过 程 中 使 用 如 下 代码 使 其 变 为 可 用 (Windows 用 
F #£ php.ini x #"php_mbstring.dll#/ fe) 。 


--enable-mbstring=LANG 
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--enable-mbstring=all 


当 mbstring 函 数 在 PHP 中 可 用 的 时 候 ， 我 们 可 以 在 php.ini 配 置 文件 中 
设置 几 个 选项 ， 以 便 正 确 地 使 用 这 些 国 数 。 在 这 些 配 置 之 后 ， 我 们 可 以 
使 用 超过 40 个 和 mbstring 相 关 的 函数 中 的 任何 一 个 来 处 理 PHP 中 的 多 字 
节 输 入 。 


这 些 隙 数 的 手册 非常 全 面 ， 是 进行 多 字 市 字符 集 和 动态 内 容 局 级 工 
作 的 推荐 读物 。 尺 官 当 你 为 了 自学 而 阅读 PHP 手 册 的 时 候 ， 我 们 推荐 阅 
谈 这 些 内 容 ， 但 是 即便 没有 阅读 它们 ， 仅 通过 本 章 的 内 容 ， 你 已 经 了 解 
得 很 好 了 。 
27.3.3 MySQL 的 配置 修改 

本 章 中 使 用 的 本 地 化 例子 并 不 需要 在 MySQL 中 进行 显 式 修改 ， 
为 这 些 例子 不 是 数据 库 有 驱动 的 。MySQL 中 默认 使 用 的 字符 集 古 ISO- 


8859-1， 但 是 ， 这 并 不 意味 着 我 们 仪 限 于 在 数据 库 中 存储 单字 节 字 符 。 
要 了 解 和 当前 语言 相关 的 MySQL 元 素 ， 请 阅读 位 于 


http://www.mysql.com/doc/en/Localisation.html 的 MySQL 手 册 条 目 。 


27.4 创建 一 个 本 地 化 页 面 结构 


在 本 市 中 ， 我 们 将 看 到 一 个 功能 完备 的 例子 。 这 是 一 个 本 地 化 的 欢 
迎 页 面 ， 它 使 用 PHP 使 得 用 户 可 以 选择 一 种 目标 语言 并 随后 接收 相应 的 
文本 。 本 市 的 目标 是 展示 一 个 例子 ， 将 这 个 脚本 中 用 到 的 字符 串 外 部 
化 ， 这 和 是 国际 化 的 特征 之 一 。 


在 这 个 脚本 中 ， 用 户 恰 好 在 基于 英语 的 Web 站 点 上 ， 但 是 也 提供 了 
在 有 用户 的 本 地 进行 浏览 的 选项 ， 包 括 瑞 语 、 德 语 和 日 语 。 这 个 过 程 中 涉 
及 如 下 3 个 元 素 。 
。 创建 并 使 用 一 个 主 文 件 来 有 发送 本 地 相关 的 标 头 信息 。 
。 创建 并 使 用 一 个 主 文件 来 显示 基于 所 选择 的 本 地 信息 。 
。 使 用 脚本 本 对。 


程序 清单 27.1 给 出 了 用 来 发 送 本 地 相关 的 标 头 信息 的 主 文 件 的 内 


程序 清单 27.1 语言 定义 文件 


1 <?php 

2 if ((tisset($ SESSION['lang'])) || (!isset($ GET['lang'])})) { 
3 $ SESSION['lang'] = "en"; 

4: $currLang = "en"; 

g } else { 

6 $currLang = $ GET['lang']; 

T $ SESSION[ lang ] = $currLang; 


is } 


9; 

1@: switch($currLang) { 

11: case "en": 

Tas define({ "CHARSET","IS0-8859-1"}; 
Nae define("LANGCODE", en ); 

14: break; 

hs 

16: case "de": 

IT: define("CHARSET","ISO-8859-1"}; 
16: define("LANGCODE", "de"); 

19: break; 

20; 

ee 全 case "ja": 

22: define( "CHARSET", "UTF-8"); 

23: define("LANGCODE", "ja"); 

24: break; 

25: 

26: default: 

27: define({ CHARSET , "ISO0-8859-1"}; 
28 : define{"LANGCODE", "en"); 

29; break; 

30: } 

Fl 


32: header( "Content-Type: text/html;charset=".CHARSET} ; 
33: header("Content-Language: ".LANGCODE) ; 
34: ?> 


程序 清单 27.1 的 第 2 行 到 第 8 行 设置 了 存储 用 户 选 择 的 语言 选项 所 需 
的 会 话 值 。 


提示 : 





下 面 的 段落 中 列 出 的 define_ lang.php 或 lang_strings.php 文 件 中 并 没有 使 用 session_start0) 函 
数 ， 因 为 这 些 文件 是 通过 主 文 件 中 的 include0 函 数 包 含 进去 的 。 我 们 稍 后 创建 的 主 文 件 ， 调 用 
了 session_startO 国 数 ， 它 将 对 这 些 包 售 文件 有 效 。 





如 果 不 和 存在 会 话 值 ， 瑞 语 本 地 设置 将 会 被 使 用 。 如 果 你 的 站 后 默认 
是 一 个 德语 站 操 ， 我 们 将 修改 这 个 文件 以 便 献 认 地 使 用 德语 本 地 设置 。 
这 个 脚本 为 下 一 个 脚本 做 准备 ， 后 者 包含 一 个 输入 选择 机 制 ， 通 过 把 


$currLang 的 值 设置 为 第 6 行 输入 的 结果 。 


第 10 行 开始 的 switch 语 句 包 含 了 几 个 case 语 句 ， 设 计 用 来 把 相应 的 
值 赋 给 常量 CHARSET 和 LANGCODE。 第 32 行 到 第 33 行 ， 在 动态 创建 和 
发 送 Content-type 和 Content-ljanguage 标 头 的 时 候 ， 第 一 次 真正 使 用 这 些 
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档 根 目录 下 。 这 个 文件 定义 了 将 在 下 一 个 脚本 中 使 用 的 两 个 常量 ， 下 一 
个 脚本 是 真正 的 显示 脚本 。 这 两 个 常量 是 CHARSET 和 LANGCODE， 分 
别 对 应 每 个 本 地 的 字符 集 和 语言 代码 。 在 显示 脚本 中 ， 这 些 常 量 用 来 创 
建 关 于 字符 集 和 语言 代码 的 、 正 确 的 META 标 记 。 尽 管 标 头 在 这 个 脚本 
中 发 送 ， 确 保 它 们 是 页 面 本 里 的 一 部 分 以 有 助 于 任何 需要 的 表单 输入 ， 
这 是 一 个 好 办 法 。 


程序 清单 27.2 创 建 了 一 个 函数 ， 它 只 是 存储 将 要 供 显 示 脚 本 使 用 的 
外 部 化 字符 串 。 这 个 例子 使 用 了 两 个 字符 串 : 一 个 用 来 欢迎 用 户 访 问 页 
ffl (WELCOME_TXT) ， 为 一 个 用 来 介绍 语言 选择 过 程 
(CHOOSE TXT) . 





程序 清单 27.2” 字符 串 定义 文件 


1: <?php 

2: function defineStrings() { 

$i switch{$ SESSION['lang']) { 

4: case "en": 

Te define{ "WELCOME TXT'","Welcome!"); 

6: define( "CHOOSE TXT", "Choose Language"); 

Pa break; 

5: 

g case "de": 

10: define { "WELCOME TXT", "Willkommen!"); 

11: define{ "CHOOSE TXT", Sprache auswählen"); 
12: break; 

To: 

14: case "ja": 

I5: define{ "WELCOME TXT","[unprintable characters]"}; 
16: define{ "CHOOSE TXT", [unprintable characters]"}; 
17: break; 

18: 

19: default: 

20; define{ "WELCOME TXT", ‘Welcome! "}); 

21: define { "CHOOSE TXT","Choose Language"); 
22: break; 

23: \ 

24: } 

25: 7> 


使 用 随 书 光 盘 中 包含 的 lang_strings.php 文 件 ， 以 使 用 这 里 无 法 显示 
的 实际 的 日 语 字 符 。 把 这 个 文件 放 入 到 Web 浏 虎 硕 的 文档 根 目 录 下 。 这 
PAETAE abate WELCOME_TXT 和 CHOOSE_TXT， 它 们 用 于 
显示 脚本 。 这 些 常 量 定 义 于 名 为 defineStringsO 的 函数 中 ， 尽 管 我 们 可 以 
很 容易 地 把 这 个 文件 编写 为 函数 结构 之 外 的 一 个 长 长 的 switch 语 铝 。 我 
直接 把 它 放 入 到 一 个 函数 是 为 了 避免 组 织 复杂 语句 ， 也 是 为 了 在 使 用 时 
示 脚 本 的 时 候 能 够 更 容 匈 说明。 


最 后 ， 到 了 创建 显示 脚本 的 时 候 。 别 筷 了 ， 国 际 化 的 一 个 关键 因素 
外 部 化 所 有 的 字符 串 ， 以 便 只 需要 使 用 一 个 主 文件 。 程 序 清单 27.3 束 
这 样 的 一 个 例子 。 
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程序 清单 27.3 ”本 地 化 的 欢迎 脚本 


<?php 

session_start(}; 

include ‘define lang.pnp' ; 

include ‘lang strings.php' ; 

definestrings(}; 

o> 

<!DOCTYPE html> 

<html lang="<?php echo LANGCODE; ?> > 

9; <head> 

10: <title><?php echo WELCOME TXT; ?></title> 

11: <meta charset="<?7php echo CHARSET; ?>" /> 

12: <body> 

13: <hi style="text-align: center;"><?php echo WELCOME TXT; ?></h1> 
14: <p style="text-align: center; font-weight: bold; > 

15: <?php echo CHOOSE TAT; ?><br/><br/> 

16: <a href="<?php echo $ SERVER['PHP SELF']."?lang=en"; ?>"> 
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17: <img src= en flag.gif" alt="English" /></a> 

18: <a href="<?php echo $ SERVER['PHP SELF']."?lang=de"; ?>"> 
19: <img src='de Tlag.gif" alt="German" /></a> 

20: <a href="<?ophp echo $ SERVER[ PHP SELF']."?lang=ja"; ?>"> 
21: <img src="ja flag.gif" alt="Japanese"/></a> 

22: </p> 

23: </body> 

24: </html> 


你 会 注意 到 ， 程 序 清单 27.3 是 一 个 基本 的 模板 ， 因 为 在 
define_lang.php 或 lang_strings.php 文 件 中 ， 所 有 和 语言 相关 的 元 系 都 外 部 
化 了 。 上 所 有 这 3 个 文件 硝 实 都 根据 所 选择 的 或 默认 的 本 地 设置 显示 了 正 
确 的 结果 。 


第 5 行 调用 了 defineStrings() 浮 数 ， 它 随后 使 得 两 个 曲 量 的 值 变 为 可 
用 ， 这 些 第 量 在 第 8 行 、 第 10 行 、 第 11 行 、 第 13 行 和 第 15 行 使 用 。 第 16 
行 到 第 18 行 旺 示 了 表示 英语 、 德 语 和 日 语 本 地 的 旗帜 ， 这 些 旗帜 是 可 以 
点 击 的 。 当 用 户 点 击 了 这 些 旗 帜 中 的 一 个 ， 该 本 地 设置 将 变 为 新 的 、 裤 
选择 的 本 地 设置 ， 并 且 所 用 的 字符 串 将 是 新 的 本 地 设置 所 对 应 的 字符 
串 。 这 些 链接 包 侣 了 lang 变 量 ， 该 变量 作为 $_GET[lang'] 传 递 到 脚本 
中 。 如 采 码 看 一 下 程序 清单 27.1 的 第 6 行 ， 将 会 看 到 它 如 何 用 来 修改 和 


用 户 偏 好 的 本 地 相关 的 设置 。 


尺 官 这 里 使 用 一 面 旗帜 来 表示 语言 选项 只 是 为 了 在 开 友 过 程 中 进行 
说 明 ， 但是， 并 不 推荐 这 种 做 法 ， 因 为 没有 一 种 本 质 的 图 形 能 够 表示 语 
言 。 例 如 ， 使 用 英国 的 旗帜 表示 英语 ， 使 用 德国 的 旗帜 表示 德语 。 瑞 语 
在 80 多 个 不 同 的 国家 都 是 官方 或 主要 语言 ， 而 德语 全 少 在 10 个 国家 和 古 官 
方 语 言 ， 没 有 一 个 旗帜 能 够 未 示 这 么 多 的 信息 。 


把 这 个 文件 保存 为 lang_selector.php 并 将 其 放置 天 Web 浏 览 器 的 文档 
根 目 录 下 。 当 第 一 次 访问 它 的 时 候 ， 将 会 看 到 如 图 27-1 所 示 的 结果 。 


(©) Welcome! 
全 C |© http:;//localhost/27/lang_selector.php bal a” qf 
T 
Welcome! 
Choose Language 


mm o 
ae] | 


图 27-1 第 一 次 浏览 语言 选择 器 
驮 认 的 是 更 语 ， 直 到 选择 另 一 种 语言 。 相 应 的 ，Welcome 和 Choose 


Language 文 本 也 以 英文 显示 。 当 用 户 点 击 德国 旗 帆 ， 他 会 看 到 图 27-2; 
当 用 户 扣 击 日 本 旗帜 ， 他 会 看 到 图 27-3。 


(©) Willkommen! 





o)| 
€ CG | © http://localhost/27/lang_selector.ohp?lanq=de | mm aa 
Willkommen! 
Sprache auswählen 
rain 
图 27-2 ”查看 德语 页 面 
(© 0g 


e> C 


Er 





© htto://localhost/27/lang_selector.php?lang=ja 


kin 2 
AK I 





图 27-3 ”浏览 日 文 页 面 


提供 Web 站 点 的 本 地 化 版 本 的 公司 和 组 织 ， 往 往 会 伦 人 很 长 的 时 间 讨 
如 何 表 示 本 地 选项 ， 包 括 旗 帜 、 国 家 的 名 字 等 等 。 这 没有 清晰 的 答 
x, {A Ze 请 记 住 ， ANB 条 成 使 用 旗帜 。 Qa] ANTE 言 选 项 ， 


这 确实 是 


是 一 


个 商业 决策 ， 但 是， 如 朱 你 经 过 了 将 字符 日、 文本 和 图 像 外 化 的 过 程 ， 
并 且 创 建 了 一 个 国际 化 的 Web 站 扣 模 板 ， 以 备 进行 本 地 化 ， 那 么 ， 本 地 
选项 的 格 却 已 经 不 是 那么 重要 的 。 


27.5 “使 用 gettext() 来 本 地 化 应 用 程序 


前 面 的 小 节 介 绍 了 应 ARARE niin VE. SIMA 
WIAA, FY Re AE TED A 2 WY PHP A 2 gettext(), 通 往 GNU gettext, 
(一 个 API) 的 一 局 门 。 


要 了 解 关 于 GNU gettext 的 更 多 信息 ， 请 访问 
http://www.gnu.org/software/gettext/ gettext.htm 。 


要 使 用 gettext 及 其 PHP 相 关 的 函数 ， 需 要 将 目录 文件 翻 详 为 特定 的 
格 却 。 这 些 文件 的 一 种 利用 的 路 平台 编辑 右 ， 是 Poedit (参见 
http://www.poedit.com/ ) 。 一 旦 创建 了 一 个 转换 目录 模板 《都 是 外 部 化 
的 字符 串 ) ， 可 以 将 该 模板 发 送 给 你 聘请 的 翻译 人 员 ， 或 者 ， 使 用 
Transifex Chttps://www.transifex.net/ ) 或 Get 
Localization (http://www.getlocalization.com/ ) 这 样 的 服务 。 有 了 完整 的 
目录 文件 ， 可 以 将 它们 放 到 Web 服 务 占 文档 根 目 录 下 的 目录 中 ， 并 且 开 
on 使 用 getteXt hy PKI A 的 过 a eae 


可 以 从 PHP Manuallyhttp://www.php.net/gettext W H J f# Al gettext bh 
KA gettext#H KI PHP RAH fa. ATA ESA BW 


e 使 用 putenv() 对 LC_ALL environment 变 量 进行 本 地 化 设置 。 

e 使 用 setlocale() 为 LC_ALL 设 置 一 个 值 ( 参 见 
http://www.php.net/setlocale ) 。 

e 使 用 bindtextdomain() 为 一 个 给 定 的 域 设 置 翻译 目录 的 位 置 〈 在 这 种 
情况 下 ， 域 指 的 是 识别 应 用 程序 的 一 个 名 字 ， 而 不 是 


www.mydomain.com 这 样 的 域名 ， 
http://www.php.net/Bindtextdomain ) 。 

e 使 用 textdomain() 设 置 和 gettext 一 起 使 用 的 默认 域 (参见 
http://www.php.net/ textdomain ) 。 

。 从 这 里 开始 ， 使 用 gettext ("some string") EX ("some string") , X 
针对 访 字 符 串 调用 gettext 翻 译 。 因 此 ， 如 采 你 有 一 个 翻译 目录 ， 负 

bt “Welcome” Hl A 48 18 FF A “Willkommen!”, Ff ALARA HI 

境 变 量 都 设置 为 适用 于 德语 ， 如 下 的 代码 将 会 输 
出 “Willkommen!”。 


echo ("Welcome"); 


一 旦 了 解 了 应 用 程序 国际 化 和 本 地 化 的 基础 知识 ， 如 果 你 打算 开发 
一 于 应 用 ， 供 使 用 多 种 语言 的 人 来 用 ， 我 建议 你 深入 了 解 一 下 基于 
gettext 的 本 地 化 框 如 和 众多 的 翻 详 服务 〈“ 除 非 你 有 很 多 的 本 地 语言 人 员 
供 你 差 直 ， 或 者 有 很 多 的 钱 可 以 用 于 翻 详 服务 ) 。 
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在 本 章 中 ， 我 们 介绍 了 国际 化 和 本 地 化 的 基本 知识 。 我 们 学 习 了 创 
建 一 个 国际 化 站 点 的 两 个 关键 : 所 有 人 字符 串 、 文 本 和 图 形 的 外 部 化 ， 以 
及 数字 、 货 币 和 日 期 格式 的 外 部 化 。 我 们 还 了 解 了 ， 不 管 是 国际 化 还 是 
本 地 化 都 并 不 等 同 于 翻译 文本 ， 翻 译 只 是 本 地 化 的 一 部 分 。 


我 们 还 学 习 了 一 些 有 天 字符 集 的 知识 : 字符 集 可 以 是 单字 节 的 或 者 
ETEN. RAIE SARS ACIS IE WA TS A RK DAE Webi tia FY LA 
TE APB APR EAN Siz ZN MCAS) E E o 


最 后 ， 我 们 创建 了 一 个 实际 的 例子 ， 来 展示 如 何 存储 一 个 本 地 相关 
的 会 话 变 量 ， 从 而 为 一 个 已 有 的 模板 确定 和 发 送 本 地 化 字符 串 。 这 个 模 
版 可 以 供 所 有 的 本 地 使 用 ， 因 为 每 个 元 又 都 是 外 部 化 的 。 另 外 ， 我 们 还 
学 习 了 使 用 应 用 程序 框架 进行 本 地 化 的 高 级 步骤 〈 使 用 PHP 的 gettext 函 
数 ) 。 


27.7 Q&A 
Q: fEPHP+, Wear. HAA Se MAH? 


A: 这 方面 有 两 个 非常 有 用 的 函数 ，number_formatO0 和 date0， 我 们 
已 经 学 习 了 date(0) 函 数 。 要 在 一 个 本 地 化 环境 中 使 用 它 ， 只 要 重新 排列 
月 份 、 日 期 和 年 份 元 素 ， 以 适应 本 地 格式 (例如 , MM-DD-YYYY, DD- 
MM-YYYY 等 等 ) 。number_format() 函 数 用 于 数字 和 货币 ， 它 根据 本 地 
选择 ， 用 逗号 、 人 句点 或 空格 来 分 组 干 位 。 请 阅读 位 于 
http://www.php.net/number_format 的 PHP 手 册 来 了 解 可 能 的 用 法 。 


27.8 ”实践 练习 


这 个 实践 练习 设计 用 来 帮助 你 预测 可 能 的 问题 、 复 习 已 经 学 过 的 知 
识 ， 并 且 开 始 把 知识 用 于 实践 。 


1. 英语 是 一 个 单字 节 语 言 还 是 多 字 节 语言 ? 日 语 呢 ? 


2 在 一 个 本 地 化 站 点 中 ， 与 字符 编码 相关 的 两 个 关键 标 头 是 什 


LZ? 


3， 除 了 文本 字符 串 ， 在 国际 化 一 个 站 点 的 时 候 还 有 其 他 哪些 内 容 


= a we 
TUR mi IE? 


Fa 
oy 


Tae, 日语 是 双 字 市 的 。 


— 


2. 带 有 字符 集 指示 器 的 Content-Language 和 Content-Type。 


3 日期、 货币 和 数字 的 格式 化 ， 是 在 国际 化 过 程 中 需要 注意 的 其 
MRA A ATOR 


EA wel 


1. 使 用 Google Translate (Bar PKA te A), ARER E 
中 创建 的 语言 定义 和 显示 文件 ， 添 加 一 些 其 他 语言 的 "Welcome!” 消 息 。 


2. 由 于 个 推荐 将 旗帜 的 图 形 用 于 表示 选择 语言 ， 请 将 本 草 示 例文 
件 中 基于 旗帜 的 语言 选项 更 改 为 其 他 更 为 合适 的 选项 。 


285 = AA XML 


= 


将 学 习 : 


企 本 草 中 ， 


小 


。 如 何 创建 一 个 基本 的 XML 文档 结构 。 

。 如 何 使 用 DOM 据 数 在 PHP 中 访问 XML。 

e u (HA SimpleXML A 20 7£ PHP # V |e] XML. 
。 如 何 使 用 JSON 数 据 。 


本 章 介 绍 如 何 通过 PHP 使 用 XML 文档 和 JSON 数 据 ， 但 并 不 是 全 面 
的 介绍 ， 否 则 需要 编写 关于 这 一 主题 的 单独 的 一 整 本 书 来 介绍 。 然 而 ， 
对 于 刚刚 接触 XML 的 人 ， 或 者 对 于 通过 PHP 操 作 XML 的 新 手 ， 以 及 接 
收 并 使 用 JSON 数 据 的 新 手 ， 一 些 函 数 比 其 他 的 更 容易 使 用 。 本 章 介绍 
两 组 这 样 的 函数 。 


28.1 什么 是 XML 


XML 源 目 这 种 语言 的 全 名 ， 即 可 扩展 标记 语言 (Extensible Markup 
Language) 。 尽 管 这 种 语言 名 字 中 有 标记 一 词 ， 但 不 要 认为 XML 和 
HTMIL 一 样 ， 因 为 ， 除 了 这 两 种 语言 部 基于 标记 对 之 外 ， 它 们 束 没 有 什 
么 相似 性 了 。XML 用 来 存储 和 交换 标记 对 之 间 的 数据 ， 而 HTML 对 于 内 
容 中 包含 了 什么 或 者 它们 的 组 织 结 构 关 心 甚 少 ， 它 唯一 的 目的 束 古 在 浏 
tit “Fl sie ZN IK EE AY 


28.1.1 基本 XML 文档 结构 


XML 文档 包含 了 两 个 主要 元 素 : 处 理 指令 和 正文 。 处 理 指令 包含 
了 XML 声明 语句 ， 以 及 想 要 添加 的 任何 处 理 指 令 和 注释 。 


提示 : 





要 了 解 XML 文 档 的 完整 定义 ， 请 参考 位 于 http:/www.w3.org/TR/REC-xml 的 XML 规范 。 


如 下 的 代码 段 是 一 个 有 效 的 处 理 指令 。 


<?xml version="1.0" ?> 
<!-- Sample XML document - -> 


Ah S Z Ja EA AG. KML PERM, WRB 
Re, APA pet, FRAR BIR, DARE. FE—-PS XML 
档 中 ， 只 有 一 个 根 元 系 。 继 续 用 图 书 的 例子 ， 这 个 元 系 可 能 叫做 
Books， 并 且 标 记 <Books></Books> 包 围 了 所 有 的 其 他 信息 。 


<Books> 


#2 POR, AMMAN oR, TAMAS. ARATE AB pl 
Ff, Rma- NEREK, BRERA PAT BSA. AME hS 
的 元 素 。 我 们 把 这 些 子 元 素 叫 做 Titte、Author 和 PublishingInfo。 但 是 ， 
出 版 信息 可 能 包含 多 个 信息 ， 我 们 需要 出 版 社 的 名 字 、 位 置 和 出 版 年 
份 。 这 不 是 问题 ， 只 需要 在 父 元 系 中 恰好 也 是 根 元 系 的 一 个 子 元 双 ) 
创建 另外 一 组 子 元 素 。 例 如 ，<PublishingInfo> 元 素 看 上 去 如 下 所 示 。 


<PublishingInfo> 
<PublisherName>=Sams PubLlishing</PublisherName> 
<PublisherCity>Indlanapolis</PublisherGity= 
<PublishedYear=2@12</PublishedYear> 
</PublishingInfo> 


综合 起 来 ， 具 有 一 个 条 日 的 一 个 示例 books.xml 文 档 如 下 所 示 。 


<?xml version="1.0" 7> 
<!--Samole XML document --> 
<Books> 
<Book> 
<Title>A Very Good Book</Title> 
<Author=Jane Doe</Author> 
<PublishingInfo> 
<PublisherName>=Sams Publishing</PublisherName> 
<PublisherCity>Indianapolis</PublisherCity> 
<PublishedYear=2012</PublishedYear> 
</PublishingInfo> 
</Book> 
</Books> 


记 住 创建 有 效 的 XML 文档 的 如 下 两 个 重要 规则 。 


e XML 区 分 大 小 写 ， 因 此 ，<Book> 和 <book> 被 看 作 是 不 同 的 元 素 。 
© 上 所 有 的 XML 标记 必须 正确 地 结束 ，XML 标 记 必 须 正 确 地 舰 上 套 ， 并 
且 不 允许 重复 标记 。 


问 books.xml 文 件 汪 加 一 些 虚 构 的 条 目 并 且 将 其 放置 到 Web 服 务 亏 的 


文档 根 目录 下 供 稍 后 的 例子 使 用 。 我 们 可 以 和 在 本 草 后 面 给 出 的 不 同 界 面 
的 例子 中 使 用 同一 个 XML 文件 。 


28.1.2 ” 何 时 应 该 使 用 XML 和 PHP 


eS Ta) LT Ta HARDO ETE, “你 想 归 使 用 的 任何 时 
IR”, (Axe, MRAP DAA, TESS RISA ho TEA Bl 


Hl, FER PIXMLIE MII A A. eK. (Ase, KEIN A 
起 来 是 什么 样 的 呢 ? 


本 革 中 的 示例 使 用 XML 来 存储 一 个 较 小 的 图 书 书目 。 想 象 一 个 以 
专 有 数据 库 格 式 存 储 的 较 大 的 图 书 书目 ， 但 是 这 个 数据 库 能 够 以 XML 
格式 输出 数据 。 如 果 你 需要 拿 到 这 个 书目 ， 但 是 义 不 想 购 买 或 使 用 存储 
它 的 专 有 数据 库 ，XML 束 古 解 决 的 方法 。 数 据 的 所 有 者 可 以 将 目录 导 
入 为 XML 格 式 ， 而 你 随后 可 以 解析 它 并 显示 想 要 的 内 容 ， 只 要 使 用 本 
革 有 前 而 所 接 述 的 格式 之 一 束 可 以 了 。 


接 下 来 ， 我 们 将 学 习 使 用 两 组 不 同 的 函数 来 解析 XML 文 档 ， 它 们 
分 别 是 DOM 函 数 和 SimpleXML 函 效 ， 而 它们 都 产生 相同 的 结 末 《解析 
XML 文档 以 提供 数据 ， 以 便 在 随后 的 脚本 中 可 以 使 用 这 些 数据 ) ， 这 
两 种 方法 役 此 略 有 有 不同。 例如， 尽管 SimpleXML Až LE DOMA ŽE J 
于 使 用 ， 然 而 ， 正 如 你 在 随后 的 程序 清单 中 看 到 的 ， 当 使 用 较 大 的 文 
件 ， 由 于 在 开始 使 用 文件 之 前 ， 整 个 XML 文 档 都 会 加 载 到 内 存 中 并 进 
行 解 机 ， 因 此 ， 将 会 融 来 性 能 的 损失 。 选 择 哪 种 方法 ， 需 要 有 上 所 权衡 ， 
这 束 是 为 什么 本 草 只 是 介绍 两 组 函数 ， 而 让 你 目 己 设法 搞 清 楚 它 们 实际 
上 能 为 你 和 你 的 开发 环境 做 些 什 么 ， 以 及 不 能 做 什么 。 


28.2 12 FDOMES 20 7E PHP V XML 


目 从 PHP 477483, DOM XML 扩展 天 成 为 PHP 的 一 部 分 ， 而 在 PHP 5 
PARARE enn 于 ，PHP 的 默认 安装 中 包含 了 DOM 功 能 ， 
这 了 驶 确保 了 要 使 用 这 些 图 数 不 需 要 安装 和 配置 附加 的 库 或 扩展 。 


DOM 表 示 文 档 对 象 模型 (Document Object Model) 。 要 了 解 DOM 的 更 多 信息 ， 请 参 
考 http://www.w3.org/TR/DOM-Level-2-Core/core.html 。 


DOM 函 数 的 主要 目的 是 允许 我 们 使 用 DOM EU 
XML XPS A | HAE AS HY) DOM A 28042 DOMDocument->load(), 
根据 一 个 文件 的 内 容 来 创建 一 个 狐 的 DOM 树 。 在 创建 了 DOM 树 i 
RATE EH BE DOMPR ARREA E EES 28.1, DOM 

图 数 用 来 通 历 DOM 树 ， 并 获取 存储 的 信 以 供 稍 后 显示 。 


程序 清单 28.1 使 用 DOM 函 数 遍 历 XML 文 档 


O N nab wd hb — 


Ae) 


<?php 
$dom = new DomDocument; 
$dom->load("books.xm1") ; 


foreach ($dom->documentElement->childNodes as $books) { 
if (($books->nodeType == 1) && ($books->nodeName == "Book )) { 


foreach ($books->childNodes as $theBook) { 
if (($theBook->nodeType == 1) && 
{$theBook->nodeName == "Title"}) { 
$theBookTitle = $theBook->textContent; 


} 


if (($theBook->nodeType == 1) && 
($theBook ->nodeName == "Author")) { 
$theBookAuthor = $theBook->textContent; 


} 


if (($theBook->nodeType == 1) && 
{$theBook ->nodeName == "PublishingInfo")) { 


foreach ($theBook->childNodes as $thePublishingInfo) { 
if (($thePublishingInfo->nodeType == 1} && 
($thePublishingInfo->nodeName == "PublisherName")) { 
$theBookPublisher = $thePublishingInfo->textContent; 


} 
if (($thePublishingInfa->nodeType == 1} && 
($thePublishingInfo->nodeName == "PublishedYear")) { 
StheBookPublishedYear = 
$thePublishingInfo->textContent; 
} 
f 
} 
} 
echo " 


<p><em>" .$theBookTitle. "</em> 
by ".$theBookAuthor."<br/> 
published by ".$theBookPublisher." in ".$theBookPublishedYear."</p>"; 


unset {($theBookTitle) ; 
unset({$theBookAuthor) ; 
unset{$theBookPublisher) ; 
unset ($theBookPublishedYear} ; 


载 入 到 这 个 文档 中 。 正 如 我 们 在 随后 的 代码 行 中 所 见 到 的 那样 ， 这 个 文 
档 树 现在 可 以 通过 $dom 访 问 。 第 5 行 开 始 遇 有 历 文 档 树 的 主 循 环 ， 它 把 文 
档 中 的 每 个 节点 都 族 入 到 一 个 名 为 $books 的 数组 中 。 


第 6 行 但 找 一 个 名 为 “Book” 的 元 系 ， 并 且 如 琳 找 到 束 继 续 处 理 。 别 
kx J.» <Book> 
</Book> 标 记 对 包含 了 books xml 文 件 中 一 本 书 的 所 有 和 条目。 如 采 继 续 处 
理 ， 第 8 行 把 所 有 子 节 点 都 收集 到 一 个 名 为 gtheBook 的 数组 中 ， 并 且 ， 
第 9 行 到 第 12 行 以 及 第 14 行 到 第 17 行 的 让 语句 分 别 查 找 名 
为 “Title" 和 “Author 的 特定 和 节点， 并 且 将 其 值 放 入 到 $theBookTitle 和 
$theBookAuthor 变 量 中 以 供 稍 后 使 用 。 


第 19 行 开始 一 个 类 似 的 主语 多， 但 是 由 于 这 一 行 租 找 一 个 名 
为 “Publishing Info” 的 节点 ， 并 且 你 知道 <PublishingInfo> 
</PublishingInfo> 标 记 对 进一步 包含 了 子 而 点 集合 ， 所 以 需要 另 一 个 循 
环 结构 来 获取 下 一 个 数据 层级 中 的 信息 。 在 第 22 行 ， 找 到 了 子 市 点 并 且 
将 其 放置 到 名 为 $thePublishingInfo 的 数组 中 ， 随 后 ， 第 23 行 到 第 26 行 以 
及 第 28 行 到 第 32 行 的 主语 句 分 别 奉 找 名 
为 “PublisherName” 和 “PublishedYear” 的 特定 节点 ， 并 且 将 它们 的 值 放 入 
全 |$theBookPublisher 和 $theBookPublishedYear 变 量 中 以 供 和 后 使 用 。 


第 8 行 所 创建 的 循环 在 第 35 行 结束 后 ， 第 37 行 到 第 40 行 使 用 存储 在 
$theBookTitle、$theBookAuthor、$theBookPublisher 和 
$theBookPublishedYear 变 量 中 的 值 同 浏览 卉 显示 一 个 标记 的 字符 串 。 便 
HRSA, ERAITEN RATER CN] HHA ER A 


下 一 个 “Book” 条 目 。 


把 这 个 程序 清单 保存 为 domexample.php 并 且 将 其 放置 在 web 服务 器 
的 文档 根 目 孙 下 。 当 通过 Web 浏 览 右 碍 看 它 的 时 候 ， 应 该 看 到 图 28-1 上 所 
示 的 结果 。 


[3] localhost/28/domexample. 


全 C | © http://localhost/28/domexample.ohp Ki A ® 


A Fery Good Book by Jane Doe 
published by Sams Publishing in 2012 


An Academic Book by Anne Smith 
published by University of Calforma Press m 2011 


Some Fluff Fiction by Jambo Jones 
published by Fluffy Press m 2009 


图 28-1 {8 FYDOM eR Arpe RA Se AN IY SCS 


Ze J ffRPHPFHY802 ANDOM RRA ERIK, LTE AS HE SIO Bl 
XML 文档 并 保存 新 版 本 的 图 数 ， 请 参 疯 位 于 http:/www.php.net/dom 的 
PHP 手 册 。 


在 下 一 方 中 ， 我 们 将 使 用 同样 的 books.xml 文 件 ， 但 是 将 使 用 
SimpleXML 函 数组 来 获取 和 显示 其 值 。 


28.3 ”使 用 SimpleXML 函数 在 PHP 中 访问 XML 


SimpleXML 是 PHP 5 中 新 增加 的 一 个 函数 ， 它 在 默认 情况 下 可 以 使 
用 并 且 不 需要 任何 额外 的 安装 和 配置 步 骆 。 它 就 像 PHP 手 册 中 所 括 述 有 的 
一 样 ， 是 “转换 XML 的 一 个 非常 简 里 而 且 易 于 使 用 的 工具 集 *， 而 且 它 的 
功能 很 强大 。 


和 DOM 函 数组 包含 大 量 函 数 和 方法 不 同 ， 只 有 少数 几 个 
SimpleXML 函数 和 方法 ， 目 前 算 来 ， 只 有 7 个 。 最 基本 的 SimpleXML ø 
ee 象 ， 可 以 直接 访问 和 操作 这 个 对 象 ， 而 不 

要 特殊 的 函数 来 做 到 这 一 点 。 我 们 需要 知道 的 第 一 个 函数 吏 是 
ed a ie » ERA TLEER EAA ARN U 
下 所 示 。 


$object with data = simplexml load file("somefile.xml"}; 


在 程序 清 蛙 28.2 中 ， 短 短 几 行 代码 用 来 创建 一 个 SimpleXML 对 象 ， 
然后 显示 存储 在 对 象 中 的 层级 数据 。 


程序 清单 28.2 ”使 用 SimpleXML 载 入 并 显示 数据 


: <?php 

: $theData = simplexml load file{"books.xml")}); 
: echo "<pre>"; 

: print_r($theData) ; 

: echo "</pre>"; 

?> 


aoa phon = 


在 第 2 行 ， 使 用 simple_load file0 把 books.xml 的 内 容 载 入 到 一 个 名 为 
$theData 的 对 象 中 。 在 第 4 行 ， 使 用 print_r0 函 数 来 输出 存储 在 对 象 中 的 


数据 的 一 个 可 读 版 本 ， 这 些 数 据 用 <pre></pre> 标 记 对 包围 起 来 。 


把 这 SE A dump.php 并 且 将 其 放置 在 Web 服 
务 器 的 文档 根 目 录 下 。 当 通过 Web 浏 览 器 查看 它 的 时 候 ， 应 该 看 到 图 
28-2 所 示 的 结 


ta] localhost/28/simplexml_du 
= @ | @ http://localhost/28/simplexml_dump.ohp Kh] E 起 


damp leXMLElement Object 
[ 
[Book] => Array 
[ 
[0] => SimplekMLElement Object 
[ 
[Title] => A Very Good Book 
[Author] => Jane Doe 
[PublishingInfo] => JimpleXHLElement Object 
[ 
[PublisherName) => Jams Publishing 
[PublisherCity)] => Indianapolis 
[PublishedYear) => 2012 


[1] => SimpleEML Element Object 
[ 

[Title] => Bn Academic EBook 

[Author] => Anne dmith 

[PublishingInfo] => JimplekHLElement Object 

[ 

[PublisherHame) => University of California Press 
[PublisherCaty] => Berkeley 
[PublishedYear) => 2011 


[2] => JimpleHLElement Object 
[ 

[Title] => Some Fluff Fiction 

(Author) => Jimbo Jones 

[PublishingInfo] => SJimplexXMLElement Object 

[ 

[PublisherName)] => Fluffy Press 
[PublisherCity] => New York 
[PublishedYear) => 2009 


图 28-2 ”来 自 一 个 SimpleXML 对 象 的 数据 


输出 数据 并 不 是 都 像 这 样 壮观 ， 但 是 ， 它 确实 展示 了 对 象 的 结构 ， 
这 反 过 来 使 我 们 知道 了 如 何以 层级 的 方式 访问 数据 。 例 如 ， 
simplexml_dump.php 的 得 出 显示 了 一 本 书 的 条 目 
[0] => SimpleXMLElement Object 


( 
[Title] => A Very Good Book 


[Author] => Jane Doe 
[PublishingInfo] => SimpleXMLElement Object 
( 


[PublisherName] => Sams Publishing 
[PublisherCity] => Indianapolis 
[PublishedYear] => 2012 


要 下 接 引 用 这 条 记录 ， 用 法 如 下 。 


$theData->Book 
PRAT ARIS PU UR, AR Ara. 


e $theData->Book->Title for the Title 

e $theData->Book->Author for the Author 

e $theData->Book->PublishingInfo->PublisherName for the Publisher 
Name 

e $theData->Book->PublishingInfo->PublisherCity for the Publisher City 

e $theData->Book->PublishingInfo->PublishedYear for the Published 


Year 


(A ERT) nets a A Ae MN Re Aide, IE AY S| 
用 可 能 有 些 不 同 ， 这 在 程序 清单 28.3 中 可 以 看 到 。 


程序 清单 28.3 (EH SimpleXML J —7S XML X 


<?php 
$theData = simplexml load file("books.xml"); 


1 
2 

3 

4 foreach($theData->Book as $theBook}) { 

5: $theBookTitle = $theBook->Title; 

6: $theBookAuthor = $theBook->Author; 

T $theBookPublisher = $theBook->PublishingInfo->PublisherName; 

Ge $theBookPublisherCity = $theBook->PublishingInfo->PublisherCcity; 
9: $theBookPublishedYear = $theBook ->PublishingInfo->PublishedYear; 


19: 

| echo ° 

12: <p><em>".$theBookTitle."</em> 

13: by ".$theBookAuthor."<br/> 

14: published by ".$theBookPublisher." (".$theBookPublisherCity.") 
to in ".$theBookPublishedYear."</p>"; 
16: 

17: unset($theBookTitle); 

18: unset ($theBookAuthor) ; 

19: unset (S$theBookPublisher} ; 

20: unset ($theBookPublishedyYear) ; 

a ak 

22% i 


在 第 2 行 ， 使 用 simple_load _ file0 把 books.xml 的 内 容 载 入 到 一 个 名 为 
$theData 的 对 象 中 。 在 第 4 行 ， 把 $theData->Book 的 内 容 《〈 即 所 有 单个 记 
K) 放 入 到 了 一 个 名 为 $theBook 的 数组 中 。 第 5 行 到 第 9 行 收集 了 有 共 体 元 
系 的 伍 ， 从 $theBook 的 层级 开始 ， 并 且 这 些 值 在 第 11 行 到 第 15 行 输出 。 
第 17 行 到 第 20 行 重 置 了 变量 的 值 以 供 下 一 次 循环 使 用 。 


把 这 个 程序 清单 保存 为 simplexmlexample.php 并 且 将 其 放置 在 Web 
服务 器 的 文档 根 目录 下 。 当 通过 Web 浏 览 器 查看 它 的 时 候 ， 应 该 看 到 图 
28-3 所 示 的 结果 。 


Soe) 


[a] localhost/28/simplexmlexa: 


过 C |© http://localhost/28/simplexmlexample.php Ki E ® 


A Very Good Book by Jane Doe 
published by Sams Publishing (Indianapolis) im 2012 


An Academic Book by Anne Smith 
published by University of California Press (Berkeley) in 2011 


Some Fluff Fiction by Jimbo Jones 
published by Fluffy Press (New York) m 2009 


图 28-3 EH SimpleXML AA te Ay A Se ANY SC AN 


注意 ， 输 出 看 上 去 和 程序 清单 28.1 的 输出 很 类 似 ， 并 且 ， 实 际 上 ， 
SimpleXML 示 例 只 是 我 们 前 面 见 到 的 基于 DOM 的 示例 的 一 个 简单 〈 或 
者 说 更 加 精 炼 的 ) 版 本 。 


要 了 解 天 于 PHP 中 SimpleXML 滑 数 的 更 多 信息 ， 请 参阅 位 于 
http://www.php.net/simplexml 的 PHP 手 册 。 


28.4 使 用 JSON 


JSON 表 示 JavaScript Object Notation 〈JavaScript 对 象 表示 法 ) , JX 

是 男 一 种 数据 交换 格式 〈( 像 XML 一 样 )， 对 于 人 和 机 器 来 说 ， 它 都 是 

tte aye, 由 于 其 简单 性 ，JSON 输 出 变 得 越 来 越 普及 ， 并 

能 有 一 天 会 通过 API 取 代 【〈 如 果 还 没有 友 生 这 种 情况 的 话 ) XML 用 
eel 


使 用 JSON， 我 们 可 以 有 名 / 值 对 的 一 个 集合 〈 它 采用 对 象 的 形 
式 ) ， 并 且 ， 可 以 有 值 的 一 个 有 序列 表 《〈 和 它 采 用 数组 的 形式 ) 。 如 果 我 
们 将 本 章 前 面 的 books.xml 文 件 中 的 条 目 重 新 写成 JSON 的 格式 ， 它 看 上 
去 如 下 所 示 。 


{ 
"book" :| 
{ 
"title": "A Very Good Book", 
“author : "Jane Doe", 
"publisher name": "Sams Publishing", 
“publisher city’: "Indianapolis", 
"publisher year": "2012" 
} 
] 
上 


添加 两 个 其 他 的 条 日， 将 会 得 到 一 些 JSON 格 式 的 数据 ， 如 程序 清 
单 28.4 所 示 。 


程序 清单 28.4 JSON 格 式 的 图 书 数据 


1: 4 

2s "book": [ 

3: { 

4: "title":"A Very Good Book", 

a "author": "Jane Doe", 

G: “publisher name : Sams Publishing", 
了 "publisher city": "Indianapolis", 
8: “publisher_year" : 2012 

9: Fs 

10: { 

a "title":"An Academic Book", 

Te author : Anne Smith", 

T9: "publisher_name":"University of California Press 
14: “publisher city": "Berkeley", 

Ioa "publisher _year":"2011" 

Tes a 

IZ: { 

ia "title":"Some Fluff Fiction", 
19: "author":"Jimbo Jones", 

20: "publisher_name":"Fluffy Press", 
zA E- "publisher city":"New York", 

22: "publisher_year":"2009" 

29: } 

24: ] 

25 | 


要 了 解 有 天 JSON 的 更 多 知识 ， 参 见 http://www.json.org 。 在 初次 创 
建 JSON 的 时 候 ， 一 球 有 用 的 工具 十 JSON 解 析 人 如 ， 位 于 
http://json.parser.online.fr/ ， 它 使 你 能 够 解析 文本 并 找到 (和 修正 ) 语法 
音 误 。 

一 旦 有 了 JSON 数 据 ， 你 可 以 使 用 PHP 的 json_decode() 函 数 来 接受 所 
有 格式 民 好 的 数据 ， 并 且 将 它们 转换 为 一 个 对 象 ， 驳 像 前 面 的 


SimpleXML 示 例 中 所 做 的 那样 。 程 序 清单 28.5 使 用 了 一 段 简 短 的 代码 来 
载 入 一 些 JSON 数 据 ， 并 且 显 示 了 对 象 中 存储 的 数据 的 层级 。 


程序 清单 28.5” 加载 并 显示 JSON 数 据 


: <?php 

: $theData = file get _contents("books.txt"); 
echo "<pre>"; 

print _r(json decode($theData) ); 

echo "</pre>"; 

?> 


Ooh woh 一 


第 2 行使 用 file_get_contentsO0， 将 books.txt〈 包 含 了 程序 清单 28.4 中 
的 JSON 数 据 的 一 个 文本 文件 ) 的 内 容 载 入 到 一 个 名 为 $theData 的 对 象 
中 。 在 第 4 行 ，print_r(0) 函 数 输出 了 对 象 中 所 存储 的 JSON 数 据 解 码 后 的 
一 个 可 读 版 本 ， 其 内 容 用 <pre></pre> 标 签 括 了 起 来 。 


将 这 个 列表 保存 为 json_dump.php， 并 且 将 其 放置 到 Web 服 务 器 的 文 
档 根 目 录 下 。 通 过 Web 浏 帘 右 来 浏 蝶 的 时 候 ， 应 该 会 看 到 如 图 28-4 所 示 
的 内 容 。 


[=] localhost/28,/|son_dump.p! 


€e C | © http://localhost/28/json_dump.php i E ww 


stdClass Object 
( 
[book] => Array 
| 
[0] => stdClass Object 
( 
[title] => A Very Good Book 
[author] => Jane Doe 
[publisher name] => Sams Publishing 
[publisher city] => Indianapolis 
[publisher year] => 2012 
} 


[1] => stdClass Object 
( 
[title] => An Academic Book 
[author] => Anne Smith 
[publisher name] => University of California Press 
[publisher city] => Berkeley 
[publisher year] => 2011 


m1 


) 


[2] => stdClass Ob ject 
( 
[title] => Some Fluff Fiction 


[author] => Jimbo Jones 
[publisher name] => Fluffy Press 
[publisher city] => New York 
[publisher year] => 2009 


图 28-4 数据 的 JSON 格 式 


要 创建 这 个 数据 的 一 个 格式 化 版 本 ， 可 以 像 下 和 面 这 样 访问 记录 中 的 
JUR o 
e $theData->book->title for the Title 


e $theData->book->author for the Author 
e $theData->book->publisher_name for the Publisher Name 


e $theData->book->publisher_city for the Publisher City 
e $theData->book->publisher_year for the Published Year 


正如 前 面 所 提 到 的 ， 使 用 JSON 的 最 妾 见 的 方式 ， 束 古 作为 API 的 输 
出 。 如 下 的 URL 包 含 了 一 个 示例 的 API 端 点 ， 用 于 访问 Google 搜 索 API， 
et AAS Ree: Vv 表示 API 的 版 本 (这 里 是 1.0) , RRR RA HEN] 
(这 里 征 PHP) 。 
http: //ajax.googleapis.com/ajax/services/search/web?v=1.@&q=PHP 

i Me PP 28.5 277 ECA RA, ERASER EA 
json_google_dump.php， 并 将 其 放置 到 你 的 Web 服 务 占 的 文档 根 目 录 
te 


$theData = file get contents("http://ajax.googleapis.com/ajax/services/ 
search/web?v=1.@&q=PHP" ) ; 


SMW Webi bias Axe BOM ASIN, SA Bon 28-5 Ar H 


+ 
2i 


localhost/28/json_google_« 


& © © http://ocathost/28/json_google_dump.php 


stdClass Object 
( 


{responseData)] => stdClass Object 


( 


[results] => Array 


( 


[0] => stdClass Object 


( 


) 


[GsearchResultClass] => GwebSearch 

funescapedUrl] => http://www.php.net/ 

[url] => http://www.php.net/ 

{[visibleUrl] => www.php.net 

[cacheUrl] => http://www.google.com/search?q=cache:cdtJ0E-OielJ: www.php.net 

[title] => PHP: Hypertext Preprocessor 

{titleNoFormatting] => PHP: Hypertext Preprocessor 

[content] => Server-side HTML embedded scripting language. It provides web developers 


=> stdClass Object 


( 


) 


[GsearchResultClass] => GwebSearch 

{unescapedUrl] => http://php.net/manual/en/index.php 

[url] => http://php.net/manual/en/index.php 

{visibleUrl] => php.net 

[cacheUrl] => http://www.google.com/search?q—cache:WiwFqe3yfIsJ:php.net 
[title] => PHP: PHP Manual - Manual 

[titleNoFormatting] => PHP: PHP Manual 一 Manual 

[content] => Complete and hyperlinked manual documents and explains all... 


=> stdClass Object 


( 


[GsearchResultClass] => GwebSearch 

funescapedUrl] => http://en.wikipedia.org/wiki/PHP 

furl] => http://en.wikipedia.org/wiki/PHP 

[visibleUrl] => en.wikipedia.org 

feacheliri? =s nrtne /fiw anncle romi esearch Foecrarhe:-SaRR NITTT en srilkinedia ara 


ht 


图 28-5 一 个 Google 搜 索 的 JSON 输 出 


旦 了 解 了 使 用 JSON 数 据 的 基础 知识 《在 使 用 了 PHP 中 的 
json_decode0 KZ Ja) ， 那 么 ， 操 作对 象 束 变 得 很 基础 了 ， 互 联网 上 


所 有 的 数据 都 尽 在 你 的 和 擎 握 之 中 。 


了 。 要 了 解 有 关 API 的 完整 列表 和 更 多 信息 ， 请 访问 


ProgrammableWeb， 地 址 是 http://www.programmableweb.com/ 。 


with a 





哦 ， 并 不 是 所 有 的 ， 但 已经 相当 多 





28.5 ”小 结 


这 个 简短 的 一 章 介绍 了 两 组 用 来 操作 XML (DOM R Z 
SimpleXML) 和 JSON 的 PHP 函 数 。 除 了 对 这 两 个 话题 的 简短 概览 ， 我 
们 还 给 出 了 例子 ， 分 别 使 用 这 两 组 函数 来 显示 存储 在 XML 或 JSON 中 的 
言 妃 。 本 章 的 目的 是 介绍 使 用 PHP 操 作 XML 和 JSON 的 概念 。 如 果 你 对 
于 综合 使 用 XML 和 PHP 感 兴趣 ， 也 可 以 看 看 AJAX(Asynchronous 
JavaScript and XML， 异 步 JavaScript 和 XML)J， 它 通常 使 用 PHP 来 产生 或 
修改 XML 或 JSON 数 据 ， 然 后 再 将 其 显示 给 客户 端 。 


28.6 Q&A 


Q: 既然 MySQL 已 经 是 一 区 很 好 的 (而 且 人 免费 的 ) 数据 库 ， 
为 什么 还 要 使 用 XML 来 存储 数据 呢 ? 


A: XML 不 仅 可 以 用 做 一 种 存储 方法 ， 而 且 可 以 用 作 数据 传输 的 一 
个 中 介 。 例 如 ， 你 可 以 使 用 XML 来 连接 一 个 数据 库 ， 提 取 数 据 并 将 其 
及 迹 给 只 能 解释 XML 数据 的 第 三 方 函 数 。 此 外 ， 尽 管 MySQL 硝 实 十 一 
于 很 好 的 而 且 人 免费 的 数据库， 一些 用 户 可 能 不 能 访问 MySQL 或 者 
任何 其 他 的 数据 库 ， 在 这 种 悄 况 下 ，XML 文 件 可 以 元 当 一 个 数据 库 系 
统 的 角色 。 


Q: 如 何 使 用 脚本 的 其 他 部 分 所 创建 的 数组 和 对 象 来 创建 
JSON? 


A: 如 果 想 要 生成 JSON 输 出 ， 你 只 需要 使 用 json_encode() 了 水 数 ， 它 
接受 已 有 的 数组 和 对 象 并 且 将 它们 转换 为 JSON 格 式 。 人 参见 
http://www.php.net/json_encode 了 解 详细 信息 。 


28.7 KRAJ 


这 个 实践 练习 设计 用 来 帮助 你 预测 可 能 的 问题 、 复 习 已 经 学 过 的 知 
识 ， 并 且 开 始 把 知识 用 于 实践 。 


问 容 题 


1， 一 个 有 效 的 XML 文档 的 开始 行 应 该 是 什么 ? 


2. 如 下 的 代码 把 XML 内 容 放 入 到 一 个 新 的 DOM 文 档 中 了 吗 ? 


$dom = new DomDocument; 


3. 使 用 什么 代码 来 把 一 个 名 为 my.xml 的 文件 的 内 容 载 入 到 一 个 名 
为 gmyData 的 SimpleXML 对象 中 ? 


Fa 
oy 


1. <?xml version="1.0"> 


2. 没有 ， 它 只 是 创建 了 一 个 引用 为 $dom 的 DOM 文 档 。 要 载 入 内 
容 ， 必 须 还 要 使 用 如 下 代码 。 


$dom->Load{ "books.xml"}; 


3. $myData = simplexml_load_File(“my.xml”); 


EA wel 


1. 创建 一 个 脚本 ， 将 JSON 编 码 的 图 书 数 据 格 式 化 ， 从 而 在 浏览 器 
中 生成 与 程序 清单 28.1 和 程序 清单 28.3 相 同 的 结果 。 


2. 生成 一 个 数据 的 数组 ， 手 动 地 或 者 通过 脚本 从 数据 库 获 取信 
轧 ， 然 后， 将 其 编码 为 JSJON 格 式 。 如 条 随后 创建 另 一 个 脚本 来 载 入 该 
答 出 并 格式 化 它 以 备 使 用 ， 那 么 ， 你 就 可 以 为 目 己 的 应 用 程序 创建 和 使 
用 一 个 API。 


第 6 部 分 “管理 和 优化 


Apache 性 能 调 校 和 虚拟 主机 
建立 一 个 安全 的 Web 服 务 器 
优化 和 调 校 MySQL 

软件 升级 


使 用 应 用 程序 框 染 


第 29 章 Apache 性 能 调 校 和 虚拟 主机 


= 


尔 将 学 习 : 


ERREP, 


。 操作 系 统 以 及 Apache 的 哪些 相关 设置 可 能 限制 服务 器 的 扩展 性 或 
降低 性 能 。 

。 有 关 载 入 测试 Apache 的 几 种 工具 。 

e 如 何 对 Apache 调 优 。 

e 如 何 配 置 Apache 以 检测 和 防止 来 日 铬 尸 机 的 小 

。 如 何 配 置 基 于 名 字 的 虚拟 主机 、 基 于 IP 的 虚拟 主机 ， 以 及 二 者 之 
间 的 不 同 。 

。 关于 虚拟 主机 对 DNS 的 依赖 性 。 

。 如 何 设 置 大 量 的 同样 的 虚拟 主机 。 


本 章 和 管理 相关 ， 焦 点 将 是 提高 Apache 服 务 器 的 性 能 和 可 扩展 性 。 
此 外 ， 我 们 还 将 学 习 基 于 名 字 和 其 于 IP 的 虚拟 主机 ，DNS 相 关 的 问题 以 
及 和 Web 浏 归 絮 日 里 相关 的 问题 。 本 章 还 说 明了 可 以 用 来 把 客户 机 相互 
隅 离开 的 不 同方 法 ， 以 及 相关 的 安全 性 取舍 。 


29.1 可 扩展 性 问题 


本 布 介 绍 可 扩展 性 问题 以 及 如 何 防 止 这 些 问 题 。 本 布 更 像 是 一 
个 “不 要 这 么 做 ”的 列表 ， 说 明了 可 能 会 降低 性 能 或 阻碍 服务 需 升 级 的 限 
制 性 因素 。 我 们 还 探讨 了 Apache 的 预 前 调 校 以 优化 性 能 


29.1.1 操作 系统 限制 


有 几 个 操作 系统 因素 可 能 会 妨碍 Apache 的 升级 。 这 些 因素 和 进程 创 
建 、 内 存 限 制 以 及 同时 打开 的 文件 或 连接 的 最 大 数目 等 有 头 。 


提示 : 





UNIX ulimit 命 令 使 我 们 可 以 在 每 个 进程 的 基础 上 来 设置 本 节 所 介绍 的 几 个 限制 。 请 参见 
操作 系统 文档 以 了 解 ulimit 命 令 语法 的 详细 内 容 。 


1. 进程 


Apache 提 供 了 用 来 防止 服务 右 进 程 和 线程 数目 超过 条 个 限制 的 设 
兽 。 这 些 设 首 影响 到 可 扩展 性 ， 因 为 它们 限制 了 同时 到 Web 服 务 问 的 连 
接 的 数目 ， 这 反 过 来 影响 到 我 们 可 以 同时 服务 的 访问 者 的 数目 。 


ea (Multi-Processing Module，MPM) 设置 反 过 来 
可 以 通过 OS 设置 限制 进程 和 线程 的 数量 。 如 何在 不 同 的 操作 系统 之 间 
修改 这 些 限制 ? 在 Linux 内 核 中 ， 需 要 修改 /proc/sys/kernel/threads-max 文 
件 中 定义 的 NR_TASKS。 我 们 可 以 使 用 如 下 这 条 命令 来 读 取 文件 的 内 


人 > 


容 。 


# cat /proc/sys/kKernel/threads -max 


我 们 可 以 使 用 如 下 这 条 命令 与 入 文件 。 


# echo value > /proc/sys/kKernel/ threads -max 


在 Linux 上 (不 像 大 多 数 其 他 UNIX 版 本 操作 系统 ) ， 线 程 和 进程 之 
闻 有 一 个 映射 ， 并 且 从 OS 的 角度 来 看 它们 是 相似 的 。 


在 Solaris 上 ， 这 些 参数 可 以 在 /etc/system 文 件 中 修改 。 这 些 修改 不 
ony 但 可 能 需要 重新 局 动 以 便 生 效 。 我 们 可 以 通过 修改 
max_nprocs 条 目 来 修改 进程 的 总 数 ， 而 通过 修改 maxuproc 来 改变 一 个 给 
定 用 户 所 允许 的 进程 数目 。 

2. 文件 持 述 符 

不 管 何 时 ， 一 个 进程 打开 一 个 文件 或 一 个 socket 的 时 候 ， 驶 分 配 了 
一 个 叫做 文件 描述 符 的 结构 ， 直 到 文件 天 财 。0S 限 制 了 一 个 给 定 进程 
所 能 打开 的 文件 摘 述 符 的 数据 ， 因 而 也 限制 了 Web 服务 硕 所 能 同时 拥有 
的 连接 的 数目 。 如 何 修改 这 些 设置 ， 取 诀 于 操作 系统 。 在 Linux 系 统 
上 ， 我 们 可 以 读 取 或 修改 /proc/sys/fsfile-max。 在 Solaris 系 统 上 ， 我 们 必 
须 编辑 /etc/system 文 件 中 的 rlim_fd_max 的 值 。 修 改 后 需要 重新 启动 才能 
生效 


在 http://httpd.apache.org/docs/2.2/vhosts/fd-limits.html 可 以 找到 其 他 
的 信息 。 
3. 控制 外 部 进程 


Apache 提 供 了 几 条 指令 来 控制 外 部 进程 所 使 用 的 资源 的 数量 。 这 样 


的 进程 包括 从 服务 器 以 及 通过 服务 器 端 包 含 执 行 的 程序 产生 的 CGI 脚 
Aa Aes 2A nena? Sere 
程 的 一 部 分 


提示 : 


按照 本 书 的 前 几 半 的 安 竣 说 明 进 行 ， 将 会 使 得 PHP 作 为 一 个 模块 安 疾 。 因 此 ， 这 些 指令 并 
不 适用 于 你 的 情况 ， 除 非 你 修改 目 己 的 安 汪 类 型 ， 或 者 处 在 一 个 虚拟 主机 环境 下 ， 其 中 的 PHP 
并 非 作 为 一 个 模块 安 疾 的。 然而 ， 在 后 一 种 情况 下 ， 你 很 可 能 没有 办 法 修改 这 些 指令 


如 下 的 Apache 指 令 ( 用 于 httpd.conf 中 〉 只 有 在 UNIX 上 上 可用， 并且 
随 看 系统 的 不 同 而 变化 。 


。 RLimitCPU 一 一 接受 两 个 参数 : 对 一 个 进程 所 允许 的 CPU 时 间 的 秒 
数 的 软 限制 和 硬 限 制 。 如 果 使 用 关键 词 max， 表 示 操 作 系 统 所 允许 
的 最 大 设置 。 硬 限制 是 可 选 的 ， 软 限制 可 以 在 重新 局 动 之 前 更 改 ， 
便 限 制 则 为 软 限 制 设置 指定 了 最 大 许可 值 。 

。 RLimitMem 一 一 语法 和 RLimitCPU 相 同 ， 但 是 这 一 指令 指定 了 每 个 
进程 所 使 用 的 内 存 的 数量 《以 字 节 为 单位 ) 。 

。 RLimitNProc 一 一 语法 和 RLimitCPU 相 同 ， 但 是 这 一 指令 指定 了 进 
程 的 数目 。 








这 3 条 指令 对 于 预防 屎 意 程 序 或 编写 很 甜 的 程序 超出 控制 是 很 有 各 
助 的 。 


29.1.2 ”和 性 能 相 天 的 Apache 设 置 
本 节 给 出 影 啊 性 能 的 、 不 同 的 Apache 设 置 。 


1. 文件 系统 访问 


从 资源 的 角 找 来 看 ， 访 问 磁 礁 上 的 文件 是 一 个 郧 贯 的 进程 ， 因 此 应 
该 尽量 使 服务 一 个 请 求 所 需 的 磁盘 访问 的 数 日 最 小 化 。 符 写 链 接 、per- 
directory 配 置 文件 和 内 容 协 了 商 是 影响 磁 稚 访问 数目 的 一 些 因 系 。 


。 和 从 写 链接 一 一 在 UNIX 中 ， 符 号 链接 (symbolic link， 或 symlink) 是 
指 同 男 一 个 文件 的 一 类 特殊 的 文件 。 它 使 用 UNIX 的 In 命 令 创 建 ， 
并 且 在 使 一 个 文件 出 现在 不 同 的 地 方 的 时 候 很 有 用 。 


Options 指 令 所 人 允许 的 两 个 参数 是 FollowSymLinks 和 
SymLinksIfOwnerMatch。 默 认 情 况 下 ，Apache 不 会 打开 符号 连接 ， 因 为 
它们 可 能 绕 过 安全 设置 。 因 为 ， 我 们 可 以 创建 这 样 一 个 从 写 链 接 : 从 一 
个 Web 站 点 的 公共 部 分 链接 到 一 个 受 限制 的 文件 或 目录 ， 而 文件 和 目录 
本 来 是 不 能 通过 Web 打 开 的 。 因 此 ， 也 是 默认 情况 下 ，Apache 需 要 执行 
检查 来 验证 文件 不 是 一 个 符号 链接 。 如 果 SymLinksIfOwnerMatch 存 在 ， 
并 且 如 果 创 建 了 符号 链接 的 同一 个 用 户 拥有 目标 文件 ， 它 将 打开 一 个 从 
F HET 0 


HH FR EMM i ZT BEE BR ETT BRA | FA SOF RO RA BET EK 
径 执 行 ， 它 们 可 能 会 增加 系统 人 负担。 如果 要 控制 内 容 创 建 ， 我 们 应 访问 
配置 添加 一 个 Options 为 +FollowSymLinks 的 指令 ， 并 且 避 人 免 
SymLinksIfOwnerMatch 参 数 。 按 照 这 种 方法 ， 测 试 不 会 发 和 后， 并 且 执 行 


也 不 会 受到 影 啊 。 


e per-directory 配 置 文件 一 一 正如 第 3 章 所 介绍 的 ， 可 能 有 per-directory 
配置 文件 。 这 些 文件 通常 名 为 .htaccess， 提 供 了 配置 服务 器 的 一 种 


TETA, FPL IOVPAR ATES ei EH. 


然而 ， 如 果 要 使 这 一 功能 可 用 ，Apache 必 须 在 指 同 被 请 求 文件 的 路 


径 中 的 每 个 目录 中 但 找 这 些 文件 ， 这 会 给 文件 系统 访问 市 来 沉 章 人 负担。 
如 果 你 不 需要 per-directory 配 置 文件 ， 可 以 通过 在 配置 中 添加 
AllowOverride none 来 关闭 这 一 功能 。 这 么 做 将 会 避免 因 查 找 .htaccess 文 
件 而 访问 文件 系统 造成 相关 的 性 能 损失 。 





AS WPR 好 来 提供 一 个 文件 的 
不 同 版 本 。 这 通过 使 用 特定 的 和 语言 相关 的 扩展 名 来 做 到 ， 但 是 ， 
在 这 种 情况 下 ，Apache 必 须 为 每 个 请 求 而 访问 文件 系统 ， 码 找 具 有 
这 样 的 扩展 名 的 文件 。 如 采 我 们 需要 使 用 内 容 协商 ， 确 保全 少 使 用 
一 个 类 型 映射 文件 ， 以 最 小 化 对 磁盘 的 访问 。 为 了 国际 化 目的 ， 基 
于 Apache 的 内 容 协商 的 蔡 代 方法 可 以 在 第 27 章 中 找到 。 

记分 板 文件 一 一 这 是 一 个 特殊 的 文件 ，Apache 主 进程 使 用 它 来 和 旧 
操作 系统 上 的 子 进程 通信 。 我 们 可 以 使 用 ScoreBoardFile 指 令 来 指 
定 其 位 置 ， 但 大 多 数 现代 操作 系统 都 不 需要 使 用 这 个 文件 。 如 果 需 
要 这 个 文件 ， 可 以 友 现 将 其 放置 在 一 个 RAM 和 磁盘 上 会 提高 性 能 
RAM 磁 盘 是 允许 系统 内 存 的 一 部 分 作为 一 个 文件 系统 被 访问 的 一 
种 方法 。 创 建 一 个 RAM 磁 盘 的 细节 根据 系统 而 有 所 不 同 。 








2. 网 络 和 状态 设置 


一 些 和 网 络 相关 的 Apache 选 项 可 能 会 降低 性 能 。 





HostnameLookups 当 HostnameLookups 设 置 为 on 或 double 时 ， 
次 客户 机 发 出 一 个 请 求 ，Apache 会 执行 一 个 DNS 但 找 来 捕获 客 尸 机 


WEZ. APRE RE RKE FBO VIE PS EIR, SRT 
令 的 默认 设置 是 off。 如 果 你 想 要 捕获 请 求 者 的 主机 名 ， 可 以 稍 后 用 
一 个 日 志 解 析 莫 来 处 理 请 求 日 志 ， 离 线 地 进行 而 不 是 实时 地 进行 。 
接受 机 制 一 一 Apache 可 以 使 用 不 同 的 机 制 来 控制 Apache 子 进程 如 何 
处 理 请 求 。 优 化 的 方法 取决 于 特定 的 平台 和 进程 数目 。 可 以 

在 http://httpd.apache.org/docs-2.2/misc/perf-tuning.html 找到 相关 信 

局 


这 个 栋 块 收集 有 关 服 务 器 、 连 接 和 请 求 的 统计 数 
据 ， 这 会 降低 Apache 的 速度 。 为 了 优化 性 能 ， 关 闭 这 一 模块 ， 或 者 
至 少 硼 保 ExtendedStatus 设 置 为 off，off 也 是 其 默认 设置 。 


mod status 





29.2 4% FY ApacheBench#y, A wl ix 


我 们 可 以 使 用 测试 工具 和 流量 生成 工具 来 测试 站 点 的 可 扩展 性 和 性 
能 。 有 很 多 了 两 业 的 和 开源 的 工具 可 供 使 用 ， 每 个 工具 都 具有 不 同 程 度 的 
精度 。 一 般 来 说 ， 很 难 精确 地 模拟 现实 使 用 中 的 请 求 流量 ， 因 为 访问 者 
有 不 同 的 导航 模式 ， 使 用 上 共有 人 不同 速 度 的 连接 来 访问 Internet， 如 果 伦 跨 
时 间 太 长 束 会 集 止 下 载 ， 如 果 失 去 耐心 束 会 重复 地 按 下 刷新 按钮 等 等 。 
同样 的 ， 一 些 工 具 记 录 实 际 网 络 流量 供 稍 后 的 分 析 。 


然而 ， 为 了 更 快 而 不 是 更 精确 地 看 看 有 天 服务 硕 处 理 大 流量 的 能 
的 基本 信息 ，Apache 服 务 器 提供 了 一 个 简单 而 有 用 的 载 入 名 试 工具， 吓 
做 ApacheBench 或 ab。 我 们 可 以 在 Apache 安 装 位 置 的 /bin 目 录 下 找到 它 。 

这 个 工具 使 我 们 能 够 多 次 请 求 革 个 URL 并 且 时 示 出 结 采 的 一 个 概 
要 。 如 下 的 命令 使 用 10 个 客户 机 在 给 定 的 时 间 内 同时 请 求 
www.example.comfk 4 a8 HY E 5110007. 


# fusr/local/apache2/bin/ab -n 1000 -c 10 http: //www.google.com/ 


提示 : 





如 果 我 们 调用 ab 而 不 使 用 任何 参数 ， 将 会 得 到 命令 行 选项 和 语法 的 一 个 完整 列表 。 此 
外 ， 目 标 URL 末 尾 的 崖 杠 是 需要 的 ， 除 非 指定 一 个 具体 的 页 面 。 





运行 结果 如 下 所 示 。 


This is ApacheBench, Version 2.3 <$Revision: 655654 $> 
Copyright 1996 Adam Twiss, Zeus Technology Ltd, http: / /wr.zeustech.net/ 
Licensed to The Apache Software Foundation, http://www.apache.org/ 


Benchmarking www.google.com (be patient} 
Completed 10@ requests 
Completed 200 requests 
Completed 30@ requests 
Completed 400 requests 
Completed 500 requests 
Completed 609 requests 
Completed 700 requests 
Completed 809 requests 
Completed 90@ requests 
Finished 1000 requests 


server software: gws 

Server Hostname: wyw. google.com 
Server Port: 80 

Document Path: f 

Document Length: 11955 bytes 
Concurrency Level: 18 

Time taken for tests: 50.751 seconds 
Complete requests: 1900 

Failed requests: 669 


(Connect: @, Receive: @, Length: 669, Exceptions: 0} 
Write errors: @ 


Total transferred: 12710814 bytes 

HTML transferred: 11974814 bytes 

Requests per second: 19.70 [#/sec] (mean) 

Time per request: 597.515 - (mean) 

Time per request: 50.751 - (mean, across all concurrent requests) 
Transfer rate: 244,58 [Kbytes/sec] received 


Connection Times {ms} 


min mean[+/-sd] median max 


Connect: 31 S0 11.9 46 179 
Processing: 88 454 53.8 449 893 
Waiting: 84 285 119.0 282 694 
Totali 136 504 56.5 499 850 


Percentage of the requests served within a certain time (ms) 
50% 499 


19% 521 
80% 527 
90% 540 
99% 964 
98% PAA 
99% 754 


180% 850 (longest request} 


这 些 请 求 都 是 通过 Internet 对 一 个 示例 服务 大 及 出 的 。 如 宋 我 们 融和 在 
同一 台 计 算 机 上 的 服务 占 或 者 本 地 局 域 网 中 的 一 个 服务 占 执 行 测试 ， 应 
该 会 得 到 每 秒 更 多 的 请 求 。 这 个 工具 的 输出 一 目 了 然 ， 最 相关 的 结果 束 
是 每 秒 请 求 的 次 数 以 及 服务 一 个 请 求 的 平均 时 间 。 我 们 也 可 以 看 到 所 有 
的 请 求 是 如 何在 1 秒 以 内 得 到 服务 的 。 


我 们 可 以 对 请 求 数 目 以 及 同时 请 求 的 客户 数 使 用 不 同 的 设置 ， 来 观 
RRA RENSA w ITERE F E. 


29.3 ”预先 性 能 调 校 


表面 各 节 介 绍 了 哪些 设置 可 能 会 芒 介 Apache 的 扩展 ， 下 面 介绍 一 些 
预 前 增加 服务 右 性 能 的 技术 。 


29.3.1 ”把 文件 映射 到 内 存 


正如 前 面 所 介绍 的 ， 访 问 磁 盘 会 显 阁 地 影响 性 能 。 尺 官 大 多 数 现代 
操作 系统 为 最 为 频 或 访问 的 文件 准备 了 一 个 绥 存 ，Apache 也 允许 我 们 显 
式 地 把 一 个 文件 映射 到 内 存 ， 从 而 不 必 再 访问 磁盘 。 执 行 这 一 映射 的 模 
块 是 mod_file_cache， 我 们 可 以 通过 MIMapFile 指 令 指 定 要 映射 到 内 存 的 
文件 的 一 个 列表 ， 它 们 一 起 作为 一 条 完整 的 指令 提交 给 服务 右 。Apache 
2.X 中 的 另 一 条 指令 CacheFile， 接 党 一 个 文件 列表 ， 在 局 动 的 时 候 缓 存 
文件 摘 述 全， 并 且 在 请 求 之 则 你 存 它 们 ， 从 而 节省 了 频 演 地 请 求 文 件 的 
时 间 和 资源 。 


29.3.2 ”分布 负载 


为 一 种 所 和 局 性 能 的 方法 是 在 数 个 服务 帮 之 间 分 布 人 负载 。 这 可 以 通 
如 下 几 种 方法 做 到 。 


eae 
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量 ， 使 得 它 把 早 个 服务 占 看 作 来 日 外 部 。 
。 利用 一 个 软件 负载 解决 方 末 ， 它 使 用 一 个 市 有 mod_rewrite 的 反问 代 


se 
。 通过 对 服务 硕 分 工 : ERRERA ae. AA BOC TEEN AR ar DA 


及 其 他 静态 内 容 的 服务 硕 。 例 如 ， 可 以 把 网 像 放 在 名 为 
images.example.com 的 服务 硕 上 并 且 从 主 服 务 磊 链接 它们 。 


29.3.3 ”缓存 


服务 内 容 的 最 快 方法 是 根本 不 提供 它 。 这 可 以 通过 如 下 方法 来 实 
现在 请 求 资源 的 时 候 ， 使 用 相应 的 HTTP 标 头 来 通知 客户 机 和 有 效 的 
代理 。 通 过 这 种 方法 ， 一 些 出 现在 多 个 页 面 中 的 资源 不 再 需要 频繁 地 交 
换 ， 例 如 logo 或 导航 按钮 ， 它 们 只 在 特定 的 一 段 时 间 内 传送 一 次 。 


此 外 ， 我 们 可 以 使 用 Apache 2.x 的 mod_cache 来 缓存 动态 内 容 ， 这 样 
它们 束 不 必 在 每 次 请 求 的 时 候 创 建 。 这 是 一 个 潜在 的 性 能 显著 提高 点 ， 
因为 动态 内 容 通 常 需要 访问 数据 库 、 处 理 模板 等 等 ， 这 会 占用 大 量 资 
源 。 


hea: 
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题 的 更 多 信息 ， 请 参见 位 于 http://httpd.apache.org/docs/2.4/caching.html 的 Apache Caching 
Guide. 


29.3.4 ”减少 数据 传输 
减少 服务 妖 负 和 载 的 男 一 种 方法 就 是 减少 需要 传送 到 客户 机 的 数据 
量 。 这 反 过 来 也 会 使 客户 机 的 Web 站 点 运行 得 更 快 ， 尤 其 是 对 那些 使 用 
较 慢 链接 的 站 点 。 我 们 可 以 做 一 些 事情 来 达到 这 一 目的 。 
。 减少 图 像 的 数量 。 
。 减少 图 像 的 大 小 。 


e 压缩 大 的 、 可 下 载 的 文件 。 

o 提前 压缩 静态 HTML 并 且 使 用 内 容 协商 。 

e 使 用 mod_deflate 压 缩 HIML 内 容 。 如 果 CPU 的 处 理 能 力 可 用 并 且 客 
户 机 通过 较 悍 的 链接 ， 这 将 会 很 有 用 。 内 容 将 会 快速 及 送 ， 并 且 进 
程 会 很 快 应 答 共 他 的 请 求 。 


29.3.5 ”网 络 设置 


HTTP 1.1 人 允许 通过 一 个 连接 服务 多 个 请 求 ，HTTP 1.0 通 过 keep-alive 
扩展 来 实现 这 一 点 。KeepAliveTimeout 指 令 使 得 我 们 能 够 指定 服务 器 在 
天 闭 一 个 非 活 动 的 连接 之 前 所 等 每 的 最 大 的 秒 数 。 增 加 超时 时 间 童 味 看 
我 们 可 以 增加 重用 连接 的 机 会 。 另 一 方面 ， 它 也 在 等 待 时 间 期 间 把 连接 
和 Apache 进 程 联系 起 来 ， 这 可 能 妨 但 本 章 前 面 所 讨论 的 可 扩展 性 。 


29.4 ”防止 小 


拒绝 服务 (Denial of service, DoS) 攻击 同时 使 用 大 量 的 请 求 来 淹 
没 Web 服 务 器 ， 降 低 服 务 颖 速度 或 者 阻止 访问 。 一 般 来 讲 ，DoS 攻 击 很 
难 预 防 ， 并 且 通 第 解雇 它们 的 最 有 效 的 方法 在 于 网 络 级 或 操作 系统 级 。 
一 个 例子 就 是 阻止 来 目 特 定 地 址 的 对 服务 喜 的 请 求 ， 尽 管 我 们 可 以 在 
Web 服 务 右 级 阻止 地 址 ， 但 在 网 络 防火 墙 、 路 由 颖 或 者 使 用 操作 系统 网 
络 过 滤 需 阻止 它们 将 会 更 有 效 。 


滥用 的 其 他 类 型 包括 友 送 超大 请 求 或 者 同时 打开 多 个 链接 。 我 们 可 
以 限制 请 求 的 大 小 和 超时 ， 从 而 使 攻击 效 束 最 小 化 。 默 认 的 请 求 超时 是 
300 秒 ， 但 是 ， 我 们 可 以 使 用 TimeOut 指 令 修 改 它 。 有 多 个 指令 可 以 用 来 
控制 请 求 主体 和 标 尖 的 大 小 ， 它 们 是 LimitRequestBody、 
LimitRequestFields、LimitRequestFieldSize、LimitRequestLine 利 
LimitXMLRequestBody. 
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面 的 程序 的 名 称 ， 这 些 程序 重复 地 打开 站 点 链接 的 程序 。Web 搜 索引 擎 
使 用 这 些 程序 来 扫描 Internet 上 的 Web 服 务 器 ， 下 载 其 内 容 并 索引 它们 。 
现实 生活 中 的 用 户 使 用 这 些 程 序 来 下 载 Web 站 点 的 全 部 或 者 部 分 ， 以 便 
稍 后 离线 的 时 候 浏览 。 通 常 ， 这 些 程序 行为 良好 ， 但 是 ， 有 时候 它们 可 
能 具有 攻击 性 ， 并 且 用 过 多 的 同时 连接 来 淹没 你 的 网 站 ， 或 者 变 成 恶性 
NEF 
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含 了 关于 如 何 访问 Web 站 点 以 及 Web 站 点 的 哪 部 分 不 能 供 它们 使 用 的 说 
明 。 这 个 文件 的 语法 可 以 在 http://www.robotstxt.org/ 找到 。 通 过 把 一 个 
格式 正确 的 robots.txt 文 件 放 置 到 Web 服 务 器 的 文档 根 目录 下 ， 我 们 可 以 
控制 蜘蛛 的 行为 。 此 外 ， 我 们 可 以 在 路 由 问 或 操作 系统 级 别 停止 请 求 。 


29.5 ”实现 虚拟 主机 


第 一 代 Web 服 务 硕 设计 用 来 处 理 单个 站 点 的 内 容 。 在 同一 机 硕 上 和 寄 
人 存 数 个 Web 站 点 的 标准 方式 ， 征 为 每 个 站 点 安 疫 和 配置 单独 的 Web 服 务 
镶 实 例 。 随 看 Intemet 规 模 增 大 ， 寄 和 存 多 个 站 点 的 种 求 也 增长 了 ， 并 且 友 
展 出 了 一 种 更 为 有 效 的 解决 方案 : 虚拟 主机 。 虚 拟 主 机 允许 Apache 的 一 
个 单独 实例 来 服务 不 同 的 站 点 ， 通 过 域名 或 者 IP 地 址 来 识别 它们 。 基 于 
IP 的 虚拟 主机 意味 看 每 个 域名 分 配 一 个 不 同 的 下 地 址 ， 基 于 名 字 的 虚拟 
EDL RAS MZ OT EIA] PIP HOSE 


Web 客 户 机 使 用 域名 系统 (domain name server system, DNS) 来 把 
主机 名 转换 为 IP 地 址 ， 或 者 进行 相反 的 转换 。 有 如 下 几 种 映射 方式 。 


。 一 对 一 一 每 个 主机 名 分 配 一 个 单个 的 、 唯 一 的 人 地 址 。 这 用 于 基 


于 IP 的 虚拟 主机 中 。 

© 一 对 多 一 一 一 个 主机 名 分 配给 数 个 下 地 址 。 这 适合 于 多 个 Apache 实 
例 服 务 同 一 个 Web 站 点 。 如 果 每 个 服务 右 安 装 在 不 同 的 机 右上 ， 束 
可 能 在 它们 之 间 平 衡 Web 流 量 ， 提 高 可 扩展 性 。 

© 多 对 一 一 一 我 们 可 以 为 数 个 主机 名 分 配 同 一 个 下地 址 。 和 客户 机 将 会 
在 请 求 中 使 用 Host: 标 头 来 指定 它 要 访问 的 Web 站 点 。 这 用 于 基于 
名 字 的 虚拟 主机 。 





当 使 用 多 对 一 的 映射 的 时 候 ， 一 个 DNS 服务 吉 通 币 可 以 配置 为 对 每 个 DNS 请 求 以 一 个 不 
同 的 下地 址 啊 应 ， 这 有 助 于 分 散人 负载 ， 这 驶 是 轮 循 DNS。 然 而 ， 如 果 你 有 机 会 使 用 一 个 负载 
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29.5.1 基于 IP 的 虚拟 主机 


最 简单 的 虚拟 主机 配置 是 为 每 个 主机 分 配 一 个 唯一 的 下 地 址 的 时 
候 。 每 个 也 映射 到 HTTP 请 求 ，Apache 负 责 把 内 容 树 分 隔 到 它们 自己 的 
VirtualHost 容 器 中 ， 如 下 上 所 示 。 


Listen 192.168.128.10:80 
Listen 192.168.129.10:80 


<VirtualHost 192.168,128.10:80> 
DocumentRoot fusr/local/apachne2/htdocs/hostt 
fother configurations specific to this hast] 
</VirtualHost> 
<VirtualHost 192.168.129.18:80@> 
DocumentRoot /usr/local/apachez2/htdocs/host2 


fother configurations specific to this host] 
</VirtualHost> 


WRIA APS 28 EY BE ELSE DocumentRoot, AhA TE 
<VirtualHost> 段 外 指定 的 全 局 设 定 将 航 使 用 。 在 前 面 的 例子 中 ， 每 个 虚 
拟 主 机 有 自己 的 DocumentRoot。 当 一 个 请 求 到 达 ，Apache 使 用 目标 IP 地 
址 来 把 请 求 导 同 相应 的 主机 。 例 如 ， 如 果 一 个 请 求 来 自 IP 
192.168.128.10，Apache 返 回来 日 /usr/local/apache2/htdocs/host1 的 文档 。 


如 果 主 机 操作 系统 没有 使 用 VirtualHost 容 器 的 名 字 解 析 一 个 IP 地 
址 ， 并 且 没 有 ServerName 指 令 ，Apache 将 在 服务 器 局 动 的 时 候 报 告 它 无 
法 把 耳 地 址 映 届 为 主机 名 。 这 个 报告 并 不 是 一 个 重要 的 错误 。Apache 仿 
将 运行 ， 但 是 这 个 错误 指示 应 该 对 DNS 配置 做 一 些 工 作 ， 以 便 Web 浏 览 
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Listen 指 令 的 绑 定 《如 采 在 DNS 中 一 个 域名 解析 为 一 个 在 机 项 上 配置 的 
IP 地 址 ， 并 且 Apache 可 以 绑 定 到 它 ) 。 


29.5.2 ”基于 名 字 的 虚拟 主机 


作为 使 用 虚拟 主机 时 减轻 对 IP 地 址 消耗 的 一 种 方法 ，HTTP/1.1 协 议 
版 本 引入 了 Host: 标 头 ， 它 使 得 浏览 器 能 够 指定 请 求 所 要 求 的 具体 主 
机 。 这 束 允 许多 个 主机 分 圣 同 一 个 单个 的 JP 地址 。 大 多 数 浏览 右 现 在 提 
供 对 HTTP/1.1 的 支持 。 


不 能 对 基于 名 称 的 虚拟 主机 使 用 SSL， 除 非 在 严格 控制 的 情况 下 。 
请 参见 http://httpd.apache.org/docs/2.4/ssl/ssl_faq.html#vhosts 以 了 解 更 多 
言 轧 。 然 而 ， 你 可 以 对 基于 耳 的 虚拟 主机 使 用 SSL 。 


一 一 


提示 : 


尽管 Host， 的 使 用 已 经 标准 化 到 了 HTTP/1.1 规 范 中 ， 但 一 些 较 旧 的 HTTP/1.0 浏 览 器 也 支 
持 这 一 标 头 。 


程序 清单 29.1 展 示 了 来 日 Mozilla Firefox 浏 览 器 的 一 组 典型 的 请 求 标 
头 。 此 外 ， 如 果 URL 和 一 个 端口 号 一 起 输入 ， 它 也 可 以 是 Host 标 头 内 容 


的 一 部 分 。 


程序 清单 29.1 请求 标 头 


GET / HTTP/1.1 

Accept: image/gif, image/x-xbitmap, image/jpeg, image/pjpeg, */* 
Accept-Language: en-us 

Accept-Encoding: gzip, deflate 

Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/535.7 (KHTML, like Gecko) 
Chrome/16.@.912.75 Safari/535.7 

Host: host1.example.com 

Connection: Keep-Alive 


Apache tii H Host: iS RCE JL EOL WT RAPHE 
享 ， 即 本 章 前 面 所 概括 的 多 对 一 的 情况 ， 因 此 ， 叫 做 基于 名 字 的 虚拟 主 
机 。 


对 于 Apache 2.4 之 前 的 版 本 ， 使 用 NameVirtualHost 指 令 可 以 指定 IP 
地 址 筑 口 的 组 合 ， 服 务 右 正 是 在 这 一 组 合 上 接受 基于 名 称 的 虚拟 主机 
请 求 的 。 在 Apache 2.2 及 更 早 的 版 本 中 ， 这 是 用 于 基于 名 称 的 虚拟 主机 


的 一 条 必需 指令 。 


提示 : 


如 果 你 使 用 Apache 2.4， 就 不 能 使 用 NameVirtualHost 指 令 。 相 反 ， 只 要 确保 在 VirtualHost 
as FLAC Y EM AIP LIK ServerName. 


程序 清单 29.2 使 得 Apache 根 据 Host 标 头 的 内 容 把 所 有 的 连接 分 配 到 
192.168.128.10。 


程序 清单 29.2 ”基于 名 字 的 虚拟 主机 


#Only use NameVirtualHost if pre-Apache 2.4 
NameVirtualHost 192.168.128.1@ 
Listen 192.168.128.10:80 
<VirtualHost 192.168.128.10> 

ServerName host1.example.com 

DocumentRoot /usr/local/apache2/htdocs/hostt 
</VirtualHost> 
<VirtualHost 192.168.128.1@> 

ServerName host2.example.com 


DocumentRoot /usr/local/apache2/htdocs/host2 
</VirtualHost> 


对 于 解析 为 192.168.128.10 的 每 个 主机 名 ，Apache 可 以 文 持 另 一 个 
基于 名 字 的 虚拟 主机 。 如 果 来 自 于 该 IP 地 址 的 请 求 所 针对 的 主机 名 没有 
包含 在 配置 文件 之 中 ， 例 如 host3.example.com，Apache 只 古 把 请 求 关联 
到 配置 文件 中 的 第 一 个 容器 ， 在 这 个 例子 中 束 古 hostl.example.com。 对 
于 没有 市 一 个 Host 标 尖 的 请 求 ， 也 适用 这 一 行为 ， 配置 文件 的 第 一 个 容 
ars LEAR A a REY Bb NA A o 
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example.com 设 置 为 默认 域名 。 在 这 种 情况 下 ， 他 可 能 把 浏览 右 导 
可 http://hostl/ ， 而 不 是 全 称 域 名 http:Whostl.example.comy/ 。Host 标 头 内 
将 只 有 host1， 而 不 是 host1.example.com。 为 了 确保 正确 的 虐 拟 主机 容 需 
得 到 请 求 ， 我 们 可 以 使 用 程序 清单 29.3 中 所 示 的 ServerAlias 指 令 。 


程序 清单 29.3 ServerAlias 指 令 


#Only use NameVirtualHost if pre-Apache 2.4 
NameVirtualHost 192.168.128.180 
Listen 192.168.128.10:80 
<VirtualHost 192.168.128.1@> 
ServerName host1.example.com 
ServerAlias hosti 
DocumentRoot /usr/local/apache2/htdocs/hostt 
</VirtualHost> 
<VirtualHost 192.168.128.1@> 
ServerName host2.example.com 
ServerAlias host2 


DocumentRoot /usr/local/apache2/htdocs/host2 
</VirtualHost> 


实际 上 ， 我 们 可 以 给 ServerAlias 一 个 空格 分 开 的 名 字 的 列表 ， 这 些 
名 字 可 能 出 现在 Host 标 头 中 ， 这 样 ， 我 们 天 不 需要 一 个 单独 的 
VirtualHost 容 器 ， 以 及 很 多 只 是 用 来 处 理 所 有 名字 变 量 的 各 见 指令 。 


HTTP 1.1 强 制 使 用 Host 标 头 。 如 果 HTTP 请 求 行 中 的 协议 版 本 是 
1.1， 请 求 必 须 带 有 一 个 Host 标 头 。 在 基于 名 字 的 虚拟 主机 的 早期 ， 把 
Host 标 头 当 作 一 种 折 中 措施 : 需要 较 少 的 IP 资 源 ， 但 是 不 能 发 送 Host 标 
头 的 遗留 浏览 器 仍然 可 以 使 用 ， 因 此 ， 无 法 访问 所 有 的 服务 器 虚拟 主 
机 。 今 天 ， 不 再 考虑 这 一 点 ， 因 为 还 在 使 用 的 这 样 的 遗留 浏览 器 已 经 为 
数 不 多 了 。 


29.5.3 KEEEMEN 
在 前 面 的 列表 中 ，DocumentRoot 指 令 亲 从 如 下 一 个 傈 单 的 模式 。 


DocumentRoot /fusr/flocal/apache2/htdocs/hostname 


其 中 hostname 束 是 用 于 虚拟 主机 的 ServerName 中 的 全 称 域名 的 主机 
名 部 分 。 对 于 只 有 少数 几 个 虚拟 主机 的 情况 ， 这 个 配置 很 好 。 但 是 ， 如 


RAB, BRASEPAT THEMED, BEAM? 这 个 配置 文件 
会 变 得 很 难 维护 。Apache 为 带 有 mod_vhost_alias 的 同样 的 虚拟 主机 提供 
了 一 种 有 效 的 解决 方 荣 。 我 们 可 以 在 VirtualDocumentRoot 指 令 中 配置 
Apache， 把 虚拟 主机 请 求 映 射 到 分 开 的 市 有 模 陈 匹配 规则 的 内 容 树 中 。 
这 一 功能 对 于 想 要 为 每 个 用 户 提 供 一 个 虚拟 主机 的 ISP 特 询 有 用 。 如 下 
的 例子 提供 了 一 个 简单 的 大 量 虚 拟 主机 配置 。 

#Only use NameVirtualHost if pre-Apache 2.4 

NameVirtualHost 192.168.128.10 


Listen 192.168,128.10:80 


VirtualDocumentRoot /usr/local/apache2/htdocs /%1 


这 个 例子 的 VirtualDocumentRoot 指 令 中 的 %1 标 记 将 由 全 称 域名 的 
第 一 部 分 亚 代 。mod_vhost_alias 指 令 拥 有 一 种 语言 用 来 把 EFQDN 组 成 部 
分 映 碳 为 文件 系统 位 置 ， 包 括 FQDN 中 的 字符 。 


如 果 我 们 删除 所 有 的 VirtualHost 容 顷 ， 并 且 把 配置 简化 为 上 述 配 
置 ， 服 务 器 将 为 创建 于 /usr/local/apache2/htdocs 目 录 下 的 任何 子 目 录 的 访 
问 请 求 服 务 。 如 果 FQDN 的 主机 名 部 分 和 一 个 子 目 孙 匹 配 ，Apache 在 将 
请 求 翻译 为 一 个 文件 系统 位 置 的 时 候 ， 将 在 那里 查找 内 容 。 


尺 官 虚拟 主机 通常 从 主 服务 此 坏 境 继承 命令 ， 但 其 中 的 一 些 ， 例 如 
Alias 指 令 ， 不 会 被 人 继承。 例如， 虚拟 主机 将 不 会 继承 如 下 的 文件 系统 映 
HT 


Alias ficons /usr/local/apachee/icons 


Options 指 令 的 FollowSymLinks 标 志 在 这 个 环境 中 也 不 可 用 。 然 而 ， 
ScriptAlias 指 令 的 另 一 种 形式 得 到 文 持 。 


如 下 展示 的 VirtualScriptAlias 指 令 把 对 /cgi-bin 下 的 任何 资源 的 请 求 
当 作 包含 的 CGI 脚 本 。 


#Only use NameVirtualHost if pre-Apache 2.4 
Name¥YirtualHost 192.168.128.186 

Listen 192.168.128.10:8@ 

VirtualDocumentRoot /usr/local/apache2/htdocs/%1/docs 
VirtualScriptAlias /usr/local/apachee/htdocs/%1/cgi-bin 


注意 ，cgi-bin 是 该 指令 的 一 个 特殊 标记 ， 将 该 目录 只 叫做 cgi 是 不 能 
工作 的 ， 必 须 是 cgi-bin。 


为 了 满足 基于 IP 的 虚拟 主机 的 需求 ， 这 些 指令 部 有 其 他 的 形式 : 
Virtual-DocumentRootIP 和 VirtualScriptAliasIP 。 


29.6 小结 


KEE A KApacheflee (FARK, ETA) Res BI Fy 
扩展 性 和 性 能 。 在 大 多 数 情况 下 ，Web 站 点 可 扩展 性 的 问题 天 系 到 动态 
内 容 的 生成 和 数据 库 访问 。 编 号 有 效 的 脚本 将 会 绥 解 这 些 领域 的 问题 。 
和 健 件 相关 的 改善 ， 例 如 融 质 量 网 卡 和 驱动 带 ， 增 加 的 内 存 以 及 磁盘 阵 
列 也 可 以 提供 增强 的 性 能 。 


关于 虚拟 主机 ，Apache 可 以 以 多 种 方式 配置 来 处 理 虚 拟 主 机 。 不 官 
你 需要 大 量 的 同样 的 虚拟 主机 ， 还 是 一 组 不 同 的 虚拟 主机 配置 ， 或 者 是 
可 用 的 IP 地 址 的 数目 受到 限制 ， 都 有 一 种 方法 可 以 为 你 的 应 用 程序 配置 
Apache。 基 于 名 字 的 虚拟 主机 是 部 童 虚 拟 主机 而 不 需要 用 尽 耳 地 址 的 一 
种 常用 技术 。 当 我 们 有 足够 的 耻 地 址 并 且 我 们 想 要 配置 太 可 能 地 简单 的 
时 候 ， 基 于 IP 的 虚拟 主机 是 为 一 种 弟 见 方法 。 男 外 ， 如 果 我 们 不 能 修改 
DNS 的 配置 ， 可 以 为 虚拟 主机 而 使 用 单独 的 端口 号 的 资源 。 


29.7 Q&A 


Q: WAY Wl se EC A Ae a AE TR? 


A: 很 多 开 肥 者 在 本 地 或 者 通过 一 个 内 部 网 络 来 测试 他 们 的 站 后 ， 
但 是 ， 如 果 在 一 个 公共 有 的 Web 站 操 运 行 ， 测 试 会 更 好 ， 因 为 许多 的 用 三 
将 能 够 通过 慢 速 连接 访问 它 。 符 试 从 一 个 拨号 账号 去 访问 你 的 网 站 ， 并 
且 确 保 页 面 载 入 足够 快 。 首 要 的 原则 和 是， 页面 应 该 在 3 秒 内 载 入 。 


Q: 如 何 把 一 个 已 有 的 基于 名 字 的 虚拟 主机 迁移 到 它 目 己 的 
机 大 上 ， 而 同时 维持 其 服务 ? 


A: 如 末 一 个 虚拟 主机 的 目标 是 移动 到 邻近 的 机 益 ， 访 机 规 在 定义 
上 不 能 具有 相同 的 卫 地 址 ， 有 一 些 额外 的 方法 可 以 采取 。 季 见 的 做 法 像 
下 面 这 样 ， 尽 官 在 具体 步 又 上 可 能 有 多 种 形 却 。 


1. 把 DNS 映射 的 存留 时 间 Ctime-to-live, TTL) 设置 为 一 个 非常 低 
的 值 。 这 会 增加 客户 机 的 主机 名 碍 找 的 频率 。 


2. 在 具有 新 的 IP 地 址 的 旧 主 机 上 ， 配 置 一 个 IP 别 名 。 


3. 配置 虚拟 主机 的 内 容 既 由 基于 名 字 的 虚拟 主机 提供 服务 ， 也 由 
基于 IP 地 址 虚拟 主机 提供 服务 。 


4， 在 对 于 旧 IP 地 址 的 虚拟 主机 的 所 有 请 求 消失 后 〈 由 于 对 DNS 组 
存 的 旧 查找 过 期 了 ) ， 可 以 迁移 服务 器 。 


Q: 我 可 以 混用 基于 IP 的 虚拟 主机 和 基于 名 字 的 虚拟 主机 
吗 ? 


A: 是 的 。 如 果 绑 定 了 多 个 了 下地 址 ， 你 可 以 用 多 种 方式 来 分 配 其 用 
法 。 一 组 基于 名 称 的 虚拟 主机 可 能 和 每 一 个 IP 关 联 ， 只 要 对 每 一 个 IP 使 
用 一 条 单独 的 NameVirtualHost 指 令 〈( 如 果 是 Apache 2.4 之 前 的 版 本 的 
话 ) ， 或 者 是 使 用 一 组 控制 的 ServerName 指 令 。 例 如 ， 一 个 卫 可 能 专门 
用 作 SSL 的 、 基 于 IP 的 虚拟 主机 ， 而 男 一 个 IP 可 能 用 于 一 组 基于 名 称 的 
虚拟 主机 。 


29.8 ”实践 练习 


这 个 实践 练习 设计 用 来 帮助 你 预测 可 能 的 问题 、 复 习 已 经 学 过 的 知 
识 ， 并 且 开 始 把 知识 用 于 实践 。 


— 


N 
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. 指出 可 能 限制 可 扩展 性 或 影响 Apache 性 能 的 Apache 设 置 。 
， 指出 可 能 限制 可 扩展 性 的 荣 些 操作 系统 设置 。 
， 指出 提高 性 能 的 某 些 方法 。 


， 一 个 VirtualHost 容 器 中 的 ServerNamedirective 是 必需 的 吗 ? 


1. 一 些 Apache 设 置 可 能 影响 到 可 扩展 性 ， 包 括 Options 指 令 的 
FollowSymLinks 和 SymLinksIfOwnerMatch 参 数 ， 局 用 per-directory 配 置 
文件 ， 主 机 名 查找 ， 拥 有 一 个 记分 板 文件 以 及 市 有 mod_status 的 统计 集 


ZS 
LJ o 


2. 一 些 操 作 系统 设置 可 能 影响 到 可 # 展 性 ， 包 括 进 程 数目 的 限 
制 、 打 开 文 件 摘 述 符 以 及 每 个 进程 允许 的 内 存 。 


3. 下 面 是 提高 性 能 的 一 些 建议 : 通过 一 个 硬件 负载 均衡 喜 或 反 回 
代理 来 进行 负载 分 布 、 数 据 压 缩 、 绥 人 存 、 把 文件 映射 到 内 存 ， 以 及 毅 态 
编 详 柑 块 。 


4， 只 有 当 使 用 基于 名 称 的 虚拟 主机 的 时 候 ， 一 个 VirtualHost 容 天 
中 的 ServerNamedirective 才 是 必需 的 。 主 机 标 头 内 容 会 和 
ServerNamedirective 的 内 容 进 行 比较 。 如 果 [ 匹 配 不 成 功 ， 会 检查 
VirtualHost 容 器 的 ServerAlias 指 令 值 是 人 否 匹 配 。 


第 30 革 LPS AEH WebikhS as 


在 本 章 ， 我 们 将 学 习 : 


e SSL/TLS PH) WOR MI JR JE AY E MER o 
。 什么 是 安全 认证 ， 以 及 如 何 创 建 和 管理 它们 。 
e 如 何在 Apache 2.2 中 激活 mod_ssl Apache 模 块 。 


本 草 介绍 如 何 建立 一 个 能 够 安全 工作 的 Apache 服 务 器 。 如 采 你 使 用 
主机 提供 商 的 一 个 共有 于 服务 空间 ， 你 残 不 必 了 解 这 些 设置 选项 ， 因 为 设 
置 一 个 安全 的 Web 服 务 器 的 过 程 已 经 由 提供 商 来 完成 ， 通 弟 不 需 你 杀 目 
动手 去 做 。 然 而 ， 如 宋 你 具有 目 己 的 服务 右 的 root 访 问 权 限 ， 本 重 将 回 
你 介绍 如 何 使 目 己 的 Web 服 务 强 更 加 安全 。 


30.1 安全 性 的 需求 


几 种 和 Internet 相 关 的 事务 都 需要 较 高 级 别 的 安全 性 。 这 包括 金融 事 
务 ， 例 如 银行 运作 和 电子 了 商务， 还 有 敏感 信息 的 任何 交换 ， 例 如 医疗 记 
录 和 公司 文档 。Internet 上 的 安全 事务 需要 如 下 3 个 主要 元 系 。 


。 机 和 密 性 一 一 如 末 你 在 传输 或 访问 诸如 信用 卡号 或 者 个 人 医疗 病历 这 
样 的 敏感 信息 ， 肯 定 个 希望 阳 生 人 能 够 获取 它们 。 

。 完整 性 一 一 传送 的 信息 必须 保证 不 会 被 外 界 修改 。 如 条 你 在 线 下 一 
个 订单 ， 购 买 100 股 的 股票 ， 不 希望 任何 人 能 够 获取 这 些 消 轧 ， 并 
且 将 订单 修改 为 购买 1000 股 。 

。 验证 一 一 需要 确信 我 们 所 通讯 的 组 织 或 者 个 人 束 古 他 们 目 己 所 表明 
的 里 份 。 


30.2 SSEL 协议 


SSL#m Secure Sockets Layer (KPERFIE) ， 而 TLS 表 示 
Transport Layer Security URMEZE) 。 这 两 个 协议 族 最 初 设计 用 来 为 
HTTP 事 务 提供 安全 性 ， 但 是 它们 也 用 于 各 种 其 他 的 Internet 协 议 ， 如 
IMAP (Internet Message Access Protocol， 互 联网 邮件 访问 协议 ) 和 
NNTP (Network News Transfer Protocol， 网 络 新 闻 传 输 协 议 ) 。 在 SSL 
上 运行 的 HTTP 叫 做 安全 HTTP。 


Netscape 在 1994 年 友 布 了 了 SSL 版 本 2， 并 且 在 1995 年 友 布 了 SSL 版 本 
3。TLS 古 一 个 IETF 标 准 ， 它 设计 用 来 把 SSL 标 准 化 为 一 个 Internet 协 
议 ， 但 是 ， 它 只 是 SSL 版 本 3 的 一 个 修改 ， 进 行 了 少量 的 功能 增加 和 清 
理 。TLS 这 个 缩 略 语 是 Microsoft 和 Netscape 在 协议 的 命名 上 进行 争论 后 
的 结 末 ， 因 为 每 家 公司 都 想 使 用 目 己 的 名 字 。 然 而 ， 这 个 名 字 并 没有 流 
传 下 来 ， 大 多 数 人 还 是 简单 地 把 这 个 协议 叫做 SSL 。 除 非 单独 声明 ， 本 
章 剩 余 的 部 分 都 把 SSL/ATLS 称 作 SSL。 


通过 在 URI (Uniform Resource Identifier， 绽 一 资源 标识 符 ) 中 用 
https 4 fe tehttp, Been Ar EE HASSLE RAI — BURBS es. SSLAY 
HTTP HIZA fm E443. 


FASEA S SSL A AER HU TED TE a HE TERE ER SE 
Tio RIIE SARS SSL DA 2S RR ae I E R N 。 


30.2.1 解决 保密 性 需求 


SSL PP DORI DN 4 BOE RR EA]. INF ET A BIE A AE, 
JEH MRA ES, thie OC NET Aa De 
恒 明 文 ， 但 是 ， 密 文 对 于 可 能 截取 它 的 每 个 人 来 说 部 是 无 法 了 解 的 。 解 

密 是 相反 的 过 程 ， 会 把 密 文 转换 回 最 初 的 明文 。 


Ms, DBE AE OR A RUSE tH. UR 
PISS FR PRES A SE RY EE A, SREY (HORT AK TT E 
(symmetric cryptography) 。 如 果 发 送 wae 和 接受 者 拥有 不 同 的 密 钥 ， 即 
补充 密 钥 ， 这 个 过 程 叫 做 非 对 称 加 密 (symmetric cryptography) IKA 4H 
加 密 (public key cryptography) 。 


1. 对 称 加 密 


如 果 用 于 加 密 和 解密 的 符 钥 是 相同 的 ， 这 个 过 程 叫 做 对 称 加 和 密 。 
DES、Triple-DES、RC4 和 RC2 都 是 用 于 对 称 密 钥 加 密 的 算法 。 大 多 数 

这 些 算法 可 以 拥有 不 同 的 密 钥 大 小 ， 以 位 为 单位 。 通 第 ， 给 定 一 个 算 
法 ， 密 钥 的 位 数 越 大 ， 加 密 越 安 全 ， 但 它 运 行 得 也 越 慢 ， 因 为 执行 拭 法 
所 需 的 计算 量 增加 了 。 


对 称 加 密 和 公 钥 加 密 相 比 要 快 一 些 ， 我 们 将 在 下 一 节 介绍 后 者 。 然 
而 ， 对 称 加 密 有 两 个 主要 的 缺陷 。 一 个 是 密 钥 必须 定期 更 改 ， 以 防止 可 
能 有 和 镭 听 者 通过 同样 的 密 钥 访问 大 量 加 密 的 材料 。 另 一 个 问题 是 密 钥 的 
分 发 问题 : AAA 而 且 是 以 安全 的 方式 
We? 这 了 驶 是 对 称 加 蜜 的 一 个 最 家 的 局 限 因 系 ， 这 个 问题 通过 定期 让 人 市 
ERA NK, TERREN. 


AI BE A DAY BR II AD TA TIE 6 FREE OT, ES 
I, “SEMA HY, MA EZRA AE. ARAYA iz, Ti 
FIT 7 ea ih BH RPA «DS BE BAAD 9 A TS EY A 
BEB Fa — A A BE o 


使 用 这 种 方法 ， 想 传送 一 条 安全 消息 给 你 的 任何 人 ， 可 以 使 用 你 的 
AHRI E AMM S RAAHAA, HWER, FRE 
RE. BUMS iT aR S AH, MEARE ETEA E. 
KERE, MAHAR ZAH, RE, HEIN A ARER 
PIKER UR ARIRE h a A H REK J I ESSE E. HA 
公 钥 的 人 将 会 把 这 些 密 钥 放 在 公 钥 服务 器 上 ， 或 者 直接 把 公 钥 发 送 给 那 
些 想 要 使 用 安全 邮件 交换 的 人 们 。 使 用 相应 的 软件 工具 ， 例 如 PGP 或 
GnuPG, RIRA AY DAA a Bee a AY ZS HRI Th ACN AS 


AAA WA 47 BERR kes AONE, eR Ss 
HU AS A AIR RAP ae I eS BU, EL aR 7 BEARS AS 23 E 
Sr FES SBR ASAE E. PATI, SUR ERIE EN SS Bee SEAR 
陷 ， 这 样 的 攻击 也 是 有 可 能 的 。 


提示 : 


公 钥 加 密 关 似 于 发送 很 多 可 交换 锁 ， 但 保留 了 主 密 铀 。 想 要 给 你 及 送 一 条 私密 消 恩 的 任 
何人 可 以 这 样 做 : 在 及 送 之 前 保证 其 安全 并 用 这 些 锁 〈 公 钥 ) 中 的 一 个 来 锁定 它 。 只 有 有 相 
MHE CAD 的 人 才能 打开 这 个 锁 〈 解 密 消 轧 ) 。 


SSL 协 议 在 最 初 的 握手 阶段 便 用 公 角 加密 来 确保 安全 地 交换 对 称 密 
钥 ， 此 后 ， 对 称 密 钥 可 以 用 来 加 密 通 讯 。 


30.2.2 ”解雇 完整 性 的 需求 


数据 完整 性 通过 如 下 方式 来 你 证 : 在 消 居 的 内 容 上 执行 一 个 特殊 的 
计算 并 且 把 结果 和 消 明 本 里 保存 起 来 。 当 消 因 到达 目 的 地 时 ， 接 受 者 可 
以 执行 相同 的 计算 并 且 比 较 结 果 。 如 来 消 明 的 内 容 肥 生 了 变化 ， 计 算 的 
结果 会 不同 ， 这 样 我 们 束 知 道 有 别 的 人 改动 了 消 明 。 


摘要 算法 也 执行 这 一 过 程 来 创建 销 息 摘要 。 针 对 任意 一 条 消息 创建 
一 个 固定 长 度 的 表示 ， 这 可 以 像 指纹 一 样 唯 一 地 标识 该 请 轧 ， 这 种 方法 
束 叫 做 消 明 摘要 (message digest) 。 一 个 好 的 消 县 摘要 算法 应 该 是 不 可 
地 的 而 且 可 以 避免 冲突 ， 人 至 少 从 实用 的 目的 是 这 样 。 


不 可 他 意 味 看 无 法 从 摘要 得 出 最 初 的 消 号 ， 而 避免 冲突 意味 看 个 会 
有 两 条 不 同 的 消 晨 拥有 相同 的 摘要 。 第 见 的 摘要 算法 古 MD5 和 SHA。 


然而 ， 只 有 消 奶 摘要 ， 无 法 你 证 消 晨 的 完整 性 ， 攻 击 者 可 以 改变 文 
本 和 消息 摘要 。 消 息 验 证 码 (message authentication code, MAC) 类 似 
于 消息 摘要 ， 但 是 在 该 过 程 中 加 入 了 一 个 共 吾 的 秘 答 密 铀 。 算 法 的 结 末 
取决 于 所 使 用 的 请 轧 和 答 铀 。 由 于 攻击 者 无 法 访问 答 铀 ， 他 无 法 同时 修 
OCA AMEH. HMAC (Hash Message Authentication Code, Hash & 
验证 人 码 ) 是 消息 验证 但 算法 的 一 个 例子 。 


SSL 协 议 使 用 MAC 代 但 来 避免 重播 攻击 ， 并 且 硝 人 了 必 达 信息 的 完 


整 性 。 
30.2.3 ”解决 验证 的 需求 


SSL 使 用 认证 来 验证 通信 中 的 团体 。 公 角 加 密 可 以 用 来 数字 化 地 标 


WKE. SEE, MOU NA SG, BC PRUE SES 
FER APU. HERAT EA BE LE SCT SE SE “Mins 2, Aa 
再 标记 该 摘要 。 


我 们 可 以 分 辨 出 创建 了 公 钥 和 私 钥 对 的 人 就 是 发 送 消 妃 的 人 ， 但 
和 是， 如何 把 密 铀 和 你 在 现实 世界 中 所 信任 的 一 个 人 或 组 织 联 系 起 来 呢 ? 
一 个 攻击 者 可 能 模仿 一 个 有 发送 着 的 号 份 并 且 友 布 一 个 不 同 的 公 铀 ， 守 称 
这 职 是 合法 的 公 铀 ， 这 也 是 有 可 能 的 。 


信任 可 以 通过 使 用 数字 证 书 来 实现 。 数 字 证 书 葡 是 包含 了 一 个 公 和 负 
和 关于 其 所 有 者 的 信息 《名 字 、 地 址 等 等 ) 的 电子 文档 。 为 了 有 用 处 ， 
证 书 几 须 由 一 个 可 信任 的 第 三 方 机 构 丛 着 《〈 即 认证 机 构 ，certification 
authority, CA) ， 认 证 机 构 来 认证 信息 古 正确 的 。 正 如 本 章 和 后 所 介绍 
的 ， 有 很 多 种 不 同 的 CA。 其 中 的 一 些 是 商业 实体 ， 为 通过 Internet 开 展 
业务 的 公司 提供 认证 服务 。 提 供 国 内 认证 服务 的 公司 创建 其 他 的 CA。 


CA 保证 证 书 中 的 信息 是 正确 的 ， 并 且 该 密 钥 属于 个 人 或 组 织 。 证 
书 也 有 一 个 有 效 时 期 ， 并 且 可 能 过 期 或 取消 。 证 书 可 以 传递 ， 因 此 ， 认 
证 过 程 可 以 委托 。 例 如 ， 一 个 可 信 的 实体 可 以 认证 公司 ， 反 过 来 ， 公 司 
可 以 负责 认证 自己 的 雇员 。 

如 果 整 个 这 个 过 程 是 有 效 的 并 且 是 可 信 的 ， 在 认证 机 构 颁 发 一 个 证 
书 之 前 ， 它 必须 要 求 有 来 目 个 人 或 组 织 的 相应 的 有 身份 证 明 。 

驮 认 情 况 下 ， 浏 览 器 包含 了 一 组 用 于 可 信 的 认证 机 构 的 根 证 书 。 
1. SSL 和 证 书 


定义 证 书 的 主要 标准 是 X.509， 适 合 于 Internet 使 用 。 一 个 X.509 证 书 
包含 以 下 信息 。 


。 他 肥 者 一 一 证 书签 名 者 的 名 字 。 
主体 一 一 你 留 认 证 的 密 钥 的 人 。 

。 主体 公 钥 一 一 主体 的 公 铀 。 

控制 信息 一 一 像 证 书 的 有 效 日 期 之 类 的 数据 。 
。 例 名 一 一 包含 前 而 的 数据 的 侈 名 。 





我 们 可 以 通过 浏 吃 占 连 接 到 一 个 安全 的 服务 颖 来 检查 一 个 真实 的 证 
书 。 如 有 果 连 接 成 功 ， 一 个 小 小 的 锁 图 标 或 者 为 一 可 见 的 线索 将 会 洪 加 到 
浏 蜗 颖 的 状态 栏 上 。 根 据 浏 只 器 的 不 同 ， 我 们 应 该 能 够 日 击 表示 的 图 标 
来 查看 有 关 SSL 连 接 和 远程 服务 右 证 书 的 信息 。 在 下 面 的 例子 中 ，SSL 
证 书 在 http:/www.amazon.com/ 可 以 但 看 到 。 我 们 可 以 看 到 证 书 的 颁 友 者 
古 GlobalSign〔 如 图 30-1 所 示 )〉 。 这 个 页 面 无 颖 地 下 载 ， 因 为 GlobalSign 
古 一 个 可 信和 的 认证 机 构 。 
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图 30-1 www.amazon.com 所 使 用 的 SSL 证 书 


在 图 30-1 所 示 的 认证 中 ，Subject 古 名 为 Amazon.com，Inc 的 一 个 实 
体 ， 位 于 美国 华盛顿 西雅图 。 这 个 实体 第 用 的 名 称 是 


WWW.dINdZON.COM o 


C 表 示 国 家 ，ST 表 示 州 ，L 表 示 地 点 ，O 表 示 组 织 ，CN 是 常用 名 
字 。 在 一 个 Web 站 点 的 证 书 例 子 中 ， 利 用 名 字 表 示 该 Web 站 点 的 全 称 域 
名 ， 这 就 是 URL 的 服务 器 名 部 分 。 在 这 个 例子 中 ， 
www.amazon.com 域 和 主机 名 “www”。 如 条 这 个 和 我 们 在 地 址 栏 输入 的 
不 一 致 ， 浏 览 器 将 会 友 出 一 个 错误 。 


如 果 CN 是 *.amazon.com， 那 么 对 于 amazon.com 域 上 的 任何 主机 ， 
SSL 验 证 都 定 有 效 的。 


2. SSL 协 议 忆 结 


我 们 已 经 看 到 了 了 SSL 如何 通 过 加 密实 现 保密 性 ， 通 过 消 因 验证 个 实 
现 完整 性 ， 以 及 通过 证 书 和 数字 签名 实现 验证 。 建 立 一 个 SSL 连 接 的 过 
程 如 下 。 


1. 用户 使 用 目 己 的 浏 斋 硕 连 接 到 远程 Web 服 务 硕 。 
2. 握手 阶段 开始 ， 浏 贤 占 和 服务 右 交 换 密 钥 和 证 书信 息 。 


3. URRAS REPA SE, Bean, efa— 


个 可 信 的 CA 和 颁发 ， 等 等 。 
4. 可 选 地 ， 服 务 需 可 以 要 求 客 户 机 也 提供 一 个 有 效 的 证 书 。 
5. 服务 器 和 客户 机 使 用 彼此 的 公 钥 来 安全 地 达成 一 个 对 称 密 铀 。 


6. 握手 阶段 结束 ， 并 且 继 续 使 用 对 称 加 密 传输 数据 。 


30.3 ”获取 和 安装 SSL 工 具 


对 SSL 的 文 持 是 由 一 个 名 为 mod_ss] 的 Apache 模 块 所 提供 的 。 这 个 模 
块 需 要 OpenSSL 库 ， 这 是 SSL/TLS 协 议 的 一 个 开源 实现 ， 而 且 还 具有 各 
种 其 他 的 加 密 算法 。OpenSSL 基 于 Eric A.Young 和 Tim J. Hudson 开 发 的 
SSLeavy 库 。 


由 于 世界 范围 内 对 字符 串 加 蜜 发 布 的 限制 以 及 知识 产权 的 保护 ， 
SSL 相 关 工 具 的 安装 随 着 平台 不 同 而 难 易 程度 不 同 。 下 面 的 小 节 提 供 了 
获取 和 安装 SSL 相 关 工 具 的 一 个 概览 。 


30.3.1 OpenSSL 


4 OpenSSL Pr fa HW ATA OCF A Bt A AY EAE http://www.openssl.org/ 
RAJ UNIX/Linux (REA) WLP Ree BL, OpenSSLaxt i Z 
和 其 他 系统 工具 的 安装 类 似 。 然 而 ，Windows 用 户 将 会 发 现 ， 目 前 还 没 
有 免费 及 布 的 预 编 详 二 进 制 代 码 可 用 。 因 此 ，Windows 用 户 必 须 目 行 纺 
译 OpenSSL 工 具 。 安 闭 了 OpenSSL 工 具 箱 之 后 ， 将 会 拥有 创建 和 操作 证 
书 和 密 钥 以 及 和 mod_ssl Apache 模 块 交 互 的 所 有 必需 的 元 素 。 


1. Windows} FAY 228 


熟悉 构建 自己 的 二 进 制 的 过 程 的 Windows 用 户 可 以 通过 OpenSSL 站 
太 有 提供 的 OpenSSL 源 代 人 码 来 日 行 完 成 。 在 Windows 上 编译 OpenSSL 的 
说 明 可 以 在 源 发 布 中 的 INSTALL.W32 文 件 中 找到 。 重 复 这 些 说 明 已 经 
RH SABA, PATO. PRAT AH, ENS Se mal él. Arig 
的 工具 是 ActiveState Perl for Windows， 以 及 如 下 的 任 一 款 C 编 译 器 。 


e Visual C++ 
e Borland C 
e GNU C (Cygwin or MinGW) 


确 傈 按照 说 明 来 选择 相应 的 编 详 项 ， 因 为 它们 彼此 有 所 区 别 。 可 以 
在 http://httpd.apacheorg/docs/2.2/platform/win_compiling.html 找到 从 
Apache 编 译 OpenSSL 的 技巧 。 


2. UNIX/Linux 用 户 的 安装 


如 果 你 运行 的 是 Linux 或 FreeBSD 的 最 近 发 布 版 本 ，OpenSSL 可 能 
经 在 你 的 系统 上 安 冯 了。 如 果 你 需要 安装 OpenSSL， 可 以 从 OpenSSL 站 
点 下 载 源 文件 。 下 载 文件 之 后 ， 解 压缩 它 并 且 cdq 到 创建 的 目录 《把 下 列 
指令 中 的 -version 符 换 成 你 目 己 特定 的 OpenSSL 当 前 版 本 ) ， 命 令 如 下 。 


# gunzip < openssl-version.tar.gz | tar xvf - 
# cd openssl-vVersion 


完整 的 安装 说 明 可 以 在 INSTALL 文 件 中 找到 。 简 而 言 之 ，config 脚 
帮 


本 将 帮助 你 编译 软件 ， 后 面 跟着 束 是 make 和 make install 过 程 。 


30.3.2 Apache 的 mod_ssl 模 块 


过 去 ，Apache 的 SSL 扩 展 必 须 单独 发 布 ， 因 为 导出 受 限制 。 现 在 ， 
mod_ssl2# xe ĝl] Y Apache 2.2， 但 只 是 作为 源 及 布 的 一 部 分 。 尽 管 这 对 于 
UNIX/Linux 用 户 不 是 问题 ， 但 是 ，Windows 用 户 将 会 发 现 必须 从 源 代 码 
编译 Apache 以 生成 mod_ssl 模 块 ，mod_ssl 没 有 在 预 编译 和 发 布 的 二 进 制 
版 本 中 发 布 。mod_ssl 模 块 依赖 于 OpenSSL 库 ， 因 此 ， 有 效 的 OpenSSL 安 
装 也 是 必需 的 。 


1， 对 于 Windows 用 户 


当下 载 预 编译 的 安 儿 二进制 文件 的 时 候 ， 确 傈 选择 文件 名 中 市 
A “openssl” WAKA. PIU, httpd-2.2.22-win32-x86-openssl-0.9.8t.ms ii 
41%} Windows Apache 2.2.22 2238 FE FF « 


在 编 与 本 书 时 ， 针 对 Apache 2.424728 Windows it Hil] 7H. A 
这 个 版 本 的 时 候 ， 它 应 该 会 遭 从 这 一 命名 惯例 。 


如 果 你 想 使 用 mod_ssl 从 源 构 建 OpenSSL 和 Apache， 请 查看 位 于 
http://httpd.apache.org/ docs/2.4/platform/win_compiling.html HJApache X% 
档 。 重 复 这 些 说 明 超 出 了 本 书 的 范围 ， 但 它们 将 会 为 你 提供 所 需 的 全 部 
信息 。 核 心 的 要 求 如 下 所 示 。 


。 安装 OpenSSL 工 具 箱 

e Microsoft Visual C++ 5.0 或 更 高 版 本 。 
e Windows 平 侣 SDK。 

e awk LA (awk、gawk 或 类 似 工 具 ) 


2. 对 于 UNIX/Linux 用 户 


第 3 革 中 使 用 的 产 友 布 应 该 已 经 包含 了 使 用 mod_ssl 所 需 的 文件 。 因 
此 ， 对 于 使 用 mod_ssl 的 UNIX/Linux 用 户 ， 只 需要 再 次 执行 配置 过 程 和 
make/make install 过 程 ， 并 江 加 如 下 内 容 作 为 配置 命令 的 一 部 分 。 


--enable-ssl --with-ssl=/usr/local/ssl/ 


ROR ZEYH i RK OpenSSL, wR EAT ARS 
EARN AS, REE ER ASP ER BRIT So oa 


态 地 把 mod_ssl 编 译 到 Apache， 可 以 通过 执行 如 下 的 命令 检查 它 是 否 存 
在 ， 该 命令 提供 一 个 编译 进 的 模块 的 列表 。 


# f/usr/local/apache2/bin/httpd -1 


提示 : 


上 述 命 令 假设 我 们 把 Apache 安 装 到 了 /usr/local/apache2 目 录 。 


如 果 mod_ssl 作 为 一 个 动态 加 载 模块 编译 ， 如 下 的 代码 行 必 须 添 加 
到 Apache 的 配置 文件 中 (httpd.conf)。 


LoadModule ssl module modules/libmodssl.so 


完成 了 对 httpd.conf 文 件 的 改变 之 后 ， 重 新 局 动 Apache 以 使 修改 生 
效 。 如 果 在 重新 局 动 之 后 答 看 error_ log，mod_ss] 将 是 服务 左 签 名 的 一 部 
J 》 如 下 所 示 。 


Apache/2.4.1 (Unix) mod ssl/2.4.1 OpenSSL/@.9.8t PHP/S.4.0 


30.4 ”管理 证 书 


在 安装 和 配置 了 OpenSSL 和 mod_ssl 之 后 ， 一 个 可 工作 的 SSL 服 务 器 
需要 实现 的 下 一 步 束 是 创建 一 个 服务 问 证 书 。 本 节 详 细 说 明 如 何 使 用 
openss] 命 令 行 工具 来 创建 和 管理 证 书 和 蜜 钥 。 如 有 果 为 一 个 电子 商务 站 点 
使 用 SSL， 加 密 将 会 全 证 客户 数据 不 航 稻 取 ， 并 且 证 书 确信 了 客户 可 以 
验证 其 号 份 正如 目 己 所 表明 的 那样 。 


提示 : 





下 面 的 例子 引用 了 命令 行程 序 openssl 的 UNIX 版 本 。 如 果 我 们 在 Windows 下 运行 ， 需 要 使 
用 openssl.exe 代 从 ， 并 且 把 例子 中 的 路 径 改 用 反 和 斜 杠 而 不 是 和 斜 杜 。 为 外 ， 如 果 我 们 安装 
OpenSSL 的 路 人 符 和 这 里 给 出 的 不 一 样 ， 和 直接 蔡 换 例子 中 的 目录 就 可 以 了 。 


30.4.1 创建 一 个 黎 钥 对 

在 创建 一 个 证 书 请 求 之 前 ， 必 须 拥 有 一 个 公 铀 / 私 钥 对 。 假 设 我们 
要 创建 的 证 书 的 全 称 域名 (FQDN) 是 www.example.com。 我 们 可 以 通 
过 执行 如 下 命令 来 创建 密 钥 。 
# openssl genrsa --des3 -out www.exanple.com.key 1024 


e genrsa 开 天 问 OpenSSL 表 示 我 们 要 生成 一 个 键 值 对 。 

。 des3 开 关 表 示 ， 私 钥 应 该 加 蜜 并且 用 一 个 客人 码 来 保护 。 
。 out 开 关 表 示 结 果 存 储 到 哪里 。 

© 1024 表 示 所 生成 的 密 钥 的 位 数 。 


调用 这 条 命令 的 结束 如 下 上 所 示 。 


Generating RSA private key, 1024 bit long modulus 
十 十 十 十 十 十 


十 十 十 十 十 十 


e is 65537 (@x10001} 
Enter pass phrase for www.example.com.key: 


正如 你 所 见 到 的 ， 要 求 我 们 提供 一 个 密码 ， 选 择 一 个 安全 的 密码 。 
密码 不 是 保护 私 钥 所 必需 的 ， 但 是 ， 无 论 何 时 当 你 想 要 启动 服务 器 的 时 
候 ， 都 会 被 询问 密码 。 


提示 : 








我 们 可 以 选择 没有 密码 你 护 的 密 铀 。 这 很 方便 ， 因 为 我 们 不 需要 在 重 司 动 的 时 候 和 输入 密 
码 ， 但 是 这 很 不 安全 ， 并 且 服 务 器 的 不 安全 也 意味 着 密 钥 的 不 安全 。 在 任何 情况 下 ， 我 们 可 
以 通过 在 生成 命令 中 管 略 -des3 开 天 或 者 通过 执行 如 下 的 命令 来 不 你 护 密 但。 








# openssl rsa -in www.example.com.key -out www.example.com.key.unsecur 
备份 www.example.com.key 文 件 是 一 个 好 主意 。 你 可 以 通过 执行 如 
下 命令 来 了 解密 钥 文件 的 内 容 。 
# openssl rsa -noout -text -in www.example.com.key 
FEA NY 70 a Gi Ja AF 2S NZ IR o 
30.4.2 ”创建 一 个 证 书签 友 请 求 


要 获得 CA 颁发 的 一 个 证 书 ， 你 必须 提交 一 个 证 书签 友 请 求 。 要 创 
建 一 个 请 求 ， 执 行 如 下 命令 。 


# req -new -key ww.example.com.key -out www.example.com.csr 


上 述 命 令 将 会 提示 你 输入 证 书信 息 ， 如 下 上 所 示 。 


Using configuration from /fusr/local/ssl/install/openssl/openssl.cnf 
Enter PEM pass phrase: 

You are about to be asked to enter information that will be incorporated 
into your certificate request. 

What you are about to enter is what is called a Distinguished Name or a DN. 
There are quite a few fields but you can leave some blank 

For some fields there will be a default value, 

If you enter '.', the field will be left blank. 

Country Name (2 letter code) [AU]:US 

State or Province Name (full name} [Some-State]:CA 

Locality Name (eg, city} []: San Francisco 

Organization Name {eg, company} [Internet Widgits Pty Ltd]:. 
Organizational Unit Name {eg, section) []:. 

Common Name (eg, YOUR name) []: www.example.com 

Email Address []:administrator@example.com 

Please enter the following ‘extra' attributes 

to be sent with your certificate request 

A challenge password []: 

An optional company name []: 


Common Name 字 段 的 输入 要 和 你 的 Web 站 点 的 访问 者 在 他 们 的 浏 
览 需 中 输入 的 内 容 一 致 ， 这 非常 重要 。 这 是 浏览 需 对 于 远程 服务 需 证 书 
所 要 执行 的 检查 之 一 。 如 果 名 字 不 同 ， 将 会 有 一 个 警告 提示 用 户 地 址 不 
VLE. 


现在 证 书 已 存储 在 www.example.com.csr。 我 们 可 以 使 用 如 下 的 命令 
来 了 解 证 书 的 内 容 。 
# openssl req -noout -text -in www.example.com.csr 

我 们 可 以 同一 个 CA 提供 证 书签 发 请 求 以 供 处 理 。VeriSign、Thawte 
和 Network Solutions 束 是 这 样 的 CA， 当 然 还 有 很 多 CA 可 供 使 用 。 我 们 
可 以 通过 站 点 了 解 天 于 VeriSign、Thawte 和 Network Solutions H pé X FENY 
及 其 产品 的 更 多 信息 。 


e VeriSign—http:/www.verisign.com/ssl/ 


e Thawte—http://www.thawte.com/ssl/web-server-ssl-certificates/ 
e Network Solutions —http://www.networksolutions.com/SSL- 


certificates/ 
30.4.3 ”创建 一 个 目 签 友 的 证 书 

我 们 也 可 以 创建 一 个 目 签 友 的 证 书 。 也 就是 说 ， 可 以 既是 颁发 者 也 
是 证 书 的 主体 。 尽 管 这 对 于 一 个 商业 站 点 不 是 很 有 用 ， 但 它 还 是 使 我 们 
能 够 在 等 竺 来自 CA 的 正式 证 书 的 同时 ， 测 试 自己 的 mod_ssl 安 装 并 拥有 
一 个 安全 的 Web 服 务 器 。 


# openssl x509 -req -days 30 -in 
www.example.com.csr -signkey 
www.example.com.Key -out www.example.com.cert 


我 们 需要 把 证 书 www.example.com.cert〈 要 么 是 CA 所 返回 的 ， 要 么 
是 日 己任 发 的 ) 复制 到 /usr/local/ssl/openssl/certs/， 并 且 把 窗 钥 复制 


#1|/usr/local/ssl/openssl/private/ - 
通过 执行 如 下 的 命令 来 保护 密 铀 文件 。 


# chmod 400 www.example.com.key 


a) 


这 条 命令 使 得 密 钥 只 能 贷 root 用 户 阅 读 。 


30.5 SSL 配 置 


前 面 的 各 节 介 绍 了 SSL 背 后 的 〈 不 是 非 间 基础 的 ) 概念 ， 并 且 我 们 
尝 习 了 如 何 生 成 密 钥 和 和 证书。 现在， 我 们 可 以 配置 Apache 以 文 持 SSL 
了 。 正 如 在 本 章 前 面 学 过 的 ，mod_ss] 模 块 要 么 必须 静态 地 编译 ， 要 人 么 
将 其 编译 为 一 个 可 载 入 的 模块 ， 这 样 httpd.conf 文 件 中 必须 有 相应 的 
LoadModule 指 令 。 


Apache 2.2.X 市 有 一 个 “额外 的 ?配置 文件 ， 专 门 用 来 运行 一 人 台 文 持 
SSL 的 服务 器 。 要 使 用 这 个 额外 文件 ， 只 需要 去 挥 httpd.conf 中 的 如 下 这 
行 注释 ， 

Include conf/extra/nttpd-ssl.cont 


接 下 来 ， 修 改 httpd-ssl.conf 中 的 标准 配置 代码 段 。 当 然 ， 需 要 用 你 
目 己 的 信息 加 以 蔡 换 。 


UseCanonicalName On 

<VirtualHost www.example.com:443> 

ServerName www.example.cam 

SSLEngine on 

SSLCertificateFile /usr/local/ssl/openssl/certs/www.example.com.cert 
SSLCertificateKeyFile fusr/loca/ssl/openssl/certs/www.example.com.key 
</VirtualHost> 


eM Bc ST EAL, ERT 443-5 hig H 
(HTTPS 的 默认 端口 ) ， 我 们 使 用 SSLEngine 命 令 在 这 个 虚拟 主机 上 打 
开 SSL。SSLCertificateFile 和 SSLCertificateKeyfile 指 令 表 示 服 务 器 证 书 在 
哪里 以 及 包含 相关 密 钥 的 文件 在 哪里 。 


Ja SERA a8 


当 我 们 想 要 以 安全 模式 启动 Apache 的 时 候 ， 之 前 的 Apache 版 本 都 要 
求 执 行 一 条 apachectl startssl 命 令 。 然 而 ， 按 照 上 一 节 中 所 定义 的 配置 方 
E, UZER Apache MA ESSLE Apachez Ka]: 执行 
apachectl start 命 令 就 行 了。 只 要 通过 指令 把 httpd-ssl.conf 文 件 包含 在 
httpd.conf 中 ， 每 次 使 用 前 面 的 指令 ，Apache 都 会 局 动 文 持 SSL 的 服务 


BI 


AN o 


QU RARA a8 JE TEIS IT IF AT Ba ETRY GR a He RP 
的 ， 将 会 提示 我 们 输入 和 客人 码 。 在 输入 了 正确 的 密码 之 后 ，Apache 将 会 局 
动 ， 并 且 我 们 可 以 使 用 类 似 https://www.example.com/ 的 URL 安 全 地 连接 
Bt. HA, APRA CHIR. WRENN ARKDA RS a 
ky A Apache tr H AURR] ARKEEN. I, WOR BANC EA E 
Bling, WAR Apache KARA T. ETT Zl Ae E AUR 
能 绑 定 到 443 端 口 。 


30.6 ”小 结 


本 章 介绍 了 SSL 协 议和 mod _ssl 的 基础 ，mod_ss] 是 实现 对 SSL 文 持 的 
Apache 模 块 。 痛 先 介 绍 了 OpenSSL 和 mod_ssl 的 安装 和 配置 ， 以 及 如 何 
使 用 openssl 命 令 行 工具 来 实现 证 书 和 密 钥 的 生成 和 管理 。 你 可 以 访问 位 
F http://httpd.apache.org/docs/2.4/mod/mod_ssl.html 的 mod_ssl 参 考 文 档 来 
了 解 深 入 的 语法 说 明和 其 他 的 配置 信息 。 别 筷 了 ，SSL 只 是 维护 一 个 安 
全 的 服务 需 的 一 部 分 ， 后 者 还 包括 应 用 安全 补丁 、0OS 配 置 、 访 问 控 
制 、 物 理 安全 性 等 等 。 


30.7 Q&A 


Q: 我 可 以 让 基于 名 字 的 虚拟 主机 使 用 SSL 吗 ? 


A: 第 见 的 一 个 问题 是 如 何 让 基于 名 字 的 虚拟 主机 使 用 SSL。 窟 且 
是 ， 目 前 还 人 不能， 正如 第 29 章 所 介绍 的 那样 。 基 于 名 字 的 虚拟 主机 依赖 
于 HTTP 请 求 的 Host 标 头 ， 但 是 ， 证 书 验 证 则 是 发 生 在 SSL 连 接 正 在 建立 
并 且 疫 有 发 迹 HTTP 请 求 的 时 候 。 有 一 个 协议 用 来 把 已 有 的 HTTP 升 级 为 
TLS， 但 是 ， 大 多 数 当 前 浏览 器 都 不 文 持 它 〈《 人 参见 位 于 http:/www.rfc- 
editor.org/rfc/rfc2817.txt HJRFC 2817) 。 


Q: 我 可 以 和 其 他 协议 一 起 使 用 SSL 吗 ? 


A: mod_ ss] 模块 把 SSL 协 议 实 现 为 一 个 过 涛 天 。 使 用 同一 个 Apache 
服务 器 的 其 他 协议 可 以 很 容易 地 利用 SSL 的 优点 。 


30.8 ”实践 练习 


这 个 实践 练习 设计 用 来 帮助 你 复习 已 经 学 过 的 知识 。 


问答 题 


1. 说 出 在 Pnternet 上 执行 安全 通信 所 需 的 3 个 需求 。 


2. 如 何 局 动 一 个 文 持 SSL 的 Apache 实 例 ? 


oy 


fee 
1. 保密 性 、 完 整 性 和 验证 。 


2， 确 保 httpd-ssl.conf 文 件 通 过 指令 包 侣 在 httpd.conf 文 件 中 ， 并 执行 


apachectl start 命 令 。 


第 31 间 ”优化 和 调 校 MySQL 


ERE, RITKA J: 


© XTMySQLAK A ZS HY AE AS AY FEE EE LAK o 

© MySQLIKA aN KEN Ja SA, 

。 如 何 使 用 OPTIMIZE TABLE 命 令 。 

e 如 何 使 用 EXPLAIN 命 令 。 

。 如 何 使 用 FLUSH 命 令 清空 表 、 绥 存 和 日 忘 文件 。 

。 如 何 使 用 SHOW 命令 获取 有 天 数据 库 、 表 和 索引 的 信息 。 
。 如 何 使 用 SHOW 命令 获取 系统 状态 信息 。 


适当 地 关注 和 调整 MySQL 服 务 右 可 以 使 它 正 和 营 地 运行 并 且 没 有 和 意 
外 。 系 统 的 优化 包括 正确 的 使 件 维护 和 软件 调 校 。 


提示 : 


要 想 了 解 维护 和 管理 MySQL 服 务 需 的 其 他 方法 ， 考 虑 MySQL Workbench 产 品 。 可 以 
7Ehttp://www.mysql.com/products/workbench/ 找到 这 一 功能 丰富 的 图 形 界 面 的 信息 和 截屏 图 。 


31.1 构建 一 个 优化 的 平台 


设计 一 个 结构 民 好 、 规 范 化 的 数据 库 染 构 ， 只 是 优化 工作 的 一 半 


(虽然 是 重要 的 一 半 ) 。 另 外 一 半 就 是 构建 和 调 校 一 个 用 来 保存 数据 库 
的 服务 器 。 考 虑 一 个 服务 器 的 4 个 主要 部 分 : CPU. AE. BEEK aS 
和 操作 系统 。 这 些 部 分 中 的 每 一 个 都 需要 提高 速度 ， 以 及 尽 可 能 多 的 设 
计 或 编程 来 使 数据 库 更 快 。 


CPU 一 -CPU 越 快 ，MySQL 处 理 数据 也 就 越 快 。 这 没有 什么 秘 
密 ，3.0GHz 的 处 理 器 比 1.0GHz 处 理 器 显然 要 快 很 多 。 随 着 处 理 器 
速度 的 稳定 增长 ， 并 且 随 独 价格 越 来 越 合理 ， 数 据 库 很 容易 所 高 速 
度 。 

内 存 一 一 尽 可 能 地 多 使 用 RAM。 内 存 总 是 越 多 越 好 ， 而 且 RAM 现 
在 也 很 便宜 。 拥 有 足够 的 内 存 有 利于 缓解 CPU 的 低速 度 。 
ERIKAS 一 适当 的 便 盘 驱动 器 不 仅 要 足够 大 而 且 要 足够 快 ， 
才能 满足 数据 库 服 务 器 及 其 流量 。 硬 盘 驱 动 器 速度 的 一 个 重要 指标 
束 是 寻 道 时 间 ， 或 者 说 ， 张 动 器 转 一 圈 并 找到 特定 信息 段 所 需 的 时 
间 。 寻 道 时 间 以 毫秒 为 单位 ， 并 且 对 于 桌面 驱动 器 来 说 ， 平 均 寻 道 
时 间 大 约 在 8 到 9 坚 秒 ， 对 于 服务 需 来 说 ， 平 均 寻 道 时 间 大 约 在 3 到 5 
坚 秒 。 当 我 们 购买 一 个 硬盘 驱动 右 的 时 候 ， 确 保 其 大 小 足够 容纳 我 
们 了 最 终 要 存储 在 数据 库 中 的 数据 ， 并 且 速 度 足 够 快速 找到 这 些 数 
据 。 

操作 系统 一 一 如 果 我 们 使 用 的 操作 系统 资源 占用 历 害 例如 ， 
Windows) ， 那 么 有 两 种 选择 : 购买 足够 的 资源 以 供 其 使 用 ， 或 者 


使 用 那些 不 会 耗 尽 资源 的 操作 系统 。 


不 管 你 是 目 己 购买 这 些 零 部 件 来 组 疼 一 人 台 机 规 ， 还 征购 买 定 制服 务 
化 的 管理 解决 方 荣 ， 如 条 我 们 在 系统 级 把 正确 的 东西 组 合 到 一 起 ， 可 能 
只 需 几 步 殴 能 达到 全 面 的 服务 磺 优 化 。 


MySQL 引 警 的 选择 (MyISAM InnoDB) 也 是 一 个 优化 选项 。 根 
据 你 的 选择 ， 各 种 附加 的 优化 都 可 以 使 用 。 我 推荐 看 一 下 MySQL 手 册 
中 关于 特定 表 的 优化 扩 巧 ， 其 地 址 
7= http://dev.mysql.com/doc/refman/5.5/en/optimization.html ， 还 有 位 于 


http://www.mysqlperformanceblog. com/ 的 MySQL Performance 博 客 。 


1E H benchmark() ci 2 


我 们 可 以 使 用 MySQL 的 benchmarkO 函 数 对 服务 器 的 速度 进行 一 次 
快速 测试 ， 来 看 看 处 理 一 个 给 定 表达 式 需 要 多 长 时 间 。 我 们 可 以 使 用 比 
较 人 简单 的 表达 式 ， 如 10+10， 也 可 以 使 用 更 为 复杂 一 后 的 表达 式 ， 例 如 
日 期 的 提取 。 


不 管 表 达 式 的 结果 是 什么 ，benchmark0 的 结果 总 是 0。benchmark() 
的 目的 不 是 得 出 表达 式 的 结果 ， 而 是 看 看 重复 这 个 表达 式 一 定 的 次 数 需 
要 多 长 时 间 。 例 如 ， 如 下 的 命令 执行 10+10 表 达 式 100 万 次 。 


select benchmark (1000000, 190+10} ; 


这 条 命令 在 我 的 测试 系统 中 的 结 来 如 下 。 


1 row in set {@.@4sec) 


下 和 面 这 条 命令 把 日 期 近 取 表达 式 也 执行 了 100 万 炊 。 


select benchmark (10800900, extract(year from now({}}}; 


这 条 命令 在 我 的 测试 系统 中 的 结 来 如 下 。 


人 十 
| benchmark({(1@@@000, extract(year from now(}}) | 

Er a A A 二 
i a | 
+-------- ee ee AE NA E AE ee eee + 


1 row in set (@.@9sec) 


EEC ne eS EE TB a, EMTA PRI hs SY TAL 
第 一 个 测试 化 了 0.04 秒 ， 第 二 个 测 弃 化 了 0.09 秒 。 我 们 可 能 要 在 一 天 中 
不 同 的 时 间 多 次 同样 地 使 用 benchmarkO 〈 在 服务 器 处 于 不 同 负载 的 情况 
下 ) ， 从 而 对 服务 器 的 性 能 有 一 个 更 准确 的 了 解 。 并 且 ， 壬 试 其 他 一 些 
benchmark， 它 们 会 比 这 些 人 简单 示例 给 你 的 服务 右 市 来 更 大 的 处 理 压 
力 。 


31.2 ”MySQL 启动 选项 


MySQL AB 提 供 了 关于 服务 避 性 能 参数 的 丰富 信息 ， 其 中 很 多 信息 
一 般 的 用 户 肯 定 不 需要 使 用 到 。 坦 昌 讲 ， 如 条 你 在 虚拟 托 过 环境 中 使 用 
MySQL， 除 非 请 求 修 改 服务 亏 设 置 ， 售 则 你 将 无 法 使 用 这 些 信息 。 
此 ， 不 要 完全 被 这 些 信 息 所 包围 ， 本 市 包含 的 只 是 可 以 更 好 地 调 校 
MySQL 服 务 上 禹 的 一 些 常 用 局 动 选项 。 


提示 : 





可 以 在 位 于 http://dev.mysql.com/doc/refman/5.5/en/server-system-variables. html 的 MySQL 手 
有 册 阅 读 到 更 多 的 内 容 。 当 你 局 动 MySQL 的 时 候 ， 一 个 名 为 my.cnf 的 配置 文件 载 入 。 这 个 文件 
包含 了 从 奖 口 号 码 到 缓存 大 小 的 信息 ， 但 是 这 些 信 息 可 以 通过 命令 行 局 动 选 项 来 修改 。 





在 MySQL 安 装 目 录 的 suppor-files 子 目录 下 或 者 在 Windows 的 安装 目 
录 中 ， 可 以 找到 如 下 示例 配置 文件 ， 每 个 都 根据 安装 的 内 存 的 特定 范围 
进行 过 调 校 。 





e my-small.cnf 对 于 RAM 小 于 64MB 的 系统 ， 其 中 的 MySQL H Æ 


12K BEH o 


e my-medium.cnf 





FRAM/) F64MBH AZ, H PMySQLÆ A 
统 上 的 主要 服务 ， 或 者 对 于 系统 升级 到 RAM 128MB， 而 MySQL 和 
其 他 进程 分 对 内 存 。 这 是 最 和 常见 的 配置 ， 其 中 MySQL 安 状 在 作为 
一 个 Web 服 务 器 的 同一 个 系统 上 ， 并 且 接 受 一 个 中 上 度 大 小 的 流量 。 

e my-large.cnf 对 于 RAM 从 128MB 到 512MB 的 系统 ， 其 中 ， 
MySQL 是 主要 的 服务 。 





e my-huge.cnf 一 一 对 于 RAM 从 1GB 到 2GB 的 系统 ， 其 中 ，MySQL 是 
主要 的 服务 。 


要 使 用 这 些 文 件 的 任何 一 个 作为 基本 配置 文件 ， 只 需要 把 选择 的 文 
件 复制 到 /etcmy.cnf 或 my.cnf 在 你 的 系统 上 的 位 置 ， 并 且 修 改 系统 相关 
言 轧 ， 例 如 端口 或 文件 位 置 。 


关键 局 动 参数 


有 两 个 单机 有 的 局 动 参数 ， 可 能 会 对 系统 性 能 有 影 啊 很 大 ， 这 就是 
key_buffer_size 和 table_cache。 如 果 你 只 正确 地 调 校 了 两 个 服务 器 参数 ， 
请 确保 是 这 两 个 。 


key_buffer_size 的 值 束 古 用 于 索引 的 缓存 的 大 小 。 绥 存 越 大 ，SQL 
命令 完成 和 结果 返回 得 越 快 。 壬 试 在 精细 调 校 和 过 于 优化 之 间 找 到 更 好 
的 路 线 ， 我 们 可 能 在 RAM 为 512MB 的 系统 上 拥有 256MB 的 
key_buffer_size， 但 任何 超过 256MB 的 设置 可 能 导致 系统 性 能 的 下 降 。 


检查 缓存 的 实际 性 能 的 一 种 简单 方法 ， 吏 是 检查 4 个 附加 变量 : 
key_read_requests、key_reads、key_write_requests 和 key_writes。 我 们 可 
以 通过 执行 SHOW STATUS 命令 来 得 到 这 些 变 量 的 值 。 


该 命令 将 会 返回 一 个 长 长 的 变量 和 信 的 列表 ， 按 照 字 母 顺 序 排列 。 
可 以 找到 如 下 的 一 些 行 《你 的 值 可 能 有 所 不 同 ) 。 


| Key read requests | 10182771 | 
| Key reads | 9326 | 
| Key write requests | 48487 | 
| Key writes | 2287 | 


如 果 你 用 key_reads 的 值 除 以 key_read_requests 的 值 ， 结 果 应 该 小 于 
0.01. AY, WER Akey_writesH){H br b\key_write_requestsH{H, Za ARV 
B)T1. ARIMA a, Ble BUA a Rot All ze 
0.000915861721998 和 0.047167281951863， 正 好 处 在 可 接受 的 参数 范围 
内 。 我 们 可 以 尝试 通过 增加 key_buffer_size 的 值 来 使 这 些 值 变 得 更 小 ， 
但 是 ， 现 在 的 值 已 经 不 错 了。 


另 一 个 重要 的 服务 器 参数 是 table_cache， 就 是 所 有 线程 所 打开 的 表 
的 数目 。 默 认 值 是 64， 但 是 ， 我 们 可 能 根据 需要 调整 这 个 数值。 使 用 
SHOW STATUS 命令 ， 在 输出 中 找到 名 为 open_tables 的 变量 。 如 采 这 个 
数值 较 大 ， 那 么 table_cache 的 值 应 该 增加 。 


包含 在 MySQL 安 状 中 的 示例 配置 文件 使 用 了 key_buffer_size 和 和 
table_cache 的 不 同 组 合 。 我 们 可 以 使 用 这 些 组 合作 为 基础 来 做 出 任何 需 
要 的 修改 。 不 管 何 时 修改 配置 ， 都 必须 重新 局 动 服 务 套 以 使 修改 生效 。 
有 时 候 ， 还 不 知道 修改 的 结 末 是 什么 ， 在 这 种 情况 下 ， 请 确保 在 把 修改 
提交 到 产品 之 前 ， 在 开发 环境 中 笑 试 你 的 修改 。 
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优化 的 表 结 构 和 设计 良好 的 表 有 所 不 同 。 表 结构 优化 必须 在 删除 后 
重新 利用 未 使 用 的 空间 ， 并 且 在 做 出 结构 修改 之 后 对 表 进 行 基本 清理 。 
OPTIMIZE TABLE 命 令 负 责 做 到 这 些 ， 它 使 用 如 下 的 语法 。 


OPTIMIZE TABLE table name[,table name] 


例如 ， 如 果 你 想 要 优化 testDB 数 据 库 中 的 grocery_inventory *, 1E 
用 OPTIMIZE TABLE grocery_inventory。 你 可 能 会 看 到 一 条 状态 消息 
只 是 显示 “OK”， 或 者 看 到 一 条 消息 显示 “Table does not support 
optimize,doing recreate + analyze instead”。 这 两 种 情况 都 很 好 ， 因 为 其 结 
末 都 是 相同 的 ， 你 的 表 已 经 优化 了 。 


ll 


EE) 


在 进行 优化 的 时 候 ， 这 个 表 将 被 锁定 ， 因 此 ， 如 采 表 较 大 ， 应 该 在 调度 低谷 或 者 当 系 统 
流量 较 小 的 时 候 执 行 优化 。 


31.4 优化 你 的 个 询 


/人 一“ 
ZE 


得 询 优 化 和 索引 的 正确 使 用 有 很 大 关系 。EXPLAIN 命 令 检 查 一 条 
的 SELECT 语句 ， 看 它 是 否 达 到 了 最 佳 的 优化 ， 是 否 已 经 在 可 能 的 


地 方 使 用 索引 。 这 对 于 包含 JOIN 的 复杂 查询 特别 有 用 。EXPLAIN 的 语 
法 如 下 。 


EXPLAIN SELECT statement 


EXPLAIN 命 令 的 输出 是 包含 如 下 信息 列 的 一 个 表 。 











id 一 一 选择 标识 符 ID。 

select_type SELECT 语 句 的 类 型 ， 有 多 种 类 型 。 
table 表 名 。 

type 连接 类 型 ， 有 多 种 类 型 。 








这 个 列表 示 MySQL 将 使 用 哪些 索引 来 租 找 这 个 
表 中 的 行 。 如 果 结 果 十 NULL， 表 示 没 有 和 守 引 用 来 辅助 俘 询 。 那 么 
我 们 应 该 看 看 表 结 构 ， 古 侣 可 以 创建 必要 的 索引 来 提高 查询 的 性 


possible_keys 

















He o 

key 丛 询 中 实际 使 用 的 键 ， 如 采 没 有 使 用 索引 ， 为 NULL。 
key_len 使 用 的 键 的 长 度 ， 如 果 有 键 的 话 。 

ref 和 键 一 起 用 来 获取 结果 的 任何 列 。 

rows 为 执行 查询 MySQL 必 须 检 查 的 行 数 。 

extra 有 关 MySQL 将 如 何 执 行 查 询 的 附加 信息 。 有 几 个 选项 ， 


例如 Using index 〈 使 用 了 一 个 索引 ) 和 Where 〈 使 用 了 一 条 WHERE 
T: 


对 于 “选择 所 有 ?的 得 询 ， 并 没有 多 少 优化 可 人 做， 除非 还 加 一 条 使 用 
主键 的 WHERE 子 句 。possible_keys 列 随后 将 显示 PRIMARY， 而 Extra 列 
将 显示 使 用 了 Where。 


当 在 包含 JOIN 的 语句 上 使 用 EXPLAIN 的 时 候 ， 度 量 对 查询 的 优化 
的 一 种 快捷 方法 ， 就 是 得 看 rows 列 中 的 值 。 假 设 结 有 末 为 2 和 1， 把 这 些 数 
字 相 乘 ， 我 们 将 得 到 2 作为 结果 。 这 就 是 MySQL 为 了 得 到 查询 的 结果 而 
必须 得 看 的 行 数 。 我 们 和 希望 这 个 数字 尽 可 能 地 低 ， 而 2 已 经 相当 低 了 。 


要 了 解 有 关 EXPLAIN 命 令 的 更 多 信息 ， 请 参阅 位 于 
http://dev.mysql.com/doc/refman/5.0/en/ explain.html 的 MySQL 手 册 。 


31.5 ”使 用 FLUSH 命 令 


对 一 个 特定 的 数据 库 共 有 重新 加 载 权 限 的 用 户 ， 可 以 使 用 FLUSH 命 
令 来 清除 MySQL 所 使 用 的 内 部 缓存 。 通 党 ， 只 有 root 级 别 的 用 户 上 共有 执 
行 FLUSH 这 样 的 管理 性 命令 的 相应 许可 。 


FLUSH 的 语法 如 下 。 


FLUSH fiush option 
FLUSH 命 令 上 共有 9 个 个 同 的 选项 ， 最 第 用 的 选项 如 下 。 


e PRIVILEGES 
e TABLES 

e HOSTS 

e LOGS 


我 们 me vue J FLUSH PRIVILEGES 命 令 ， 在 添加 了 新 的 用 
PAE. 命令 只 是 把 MySQL 数 据 库 中 授权 的 表 重 新 载 入 ， 使 
eniin ` 需 要 停止 和 重新 启动 MySQL。 当 我 们 执行 一 条 
FLUSH PRIVILEGES 命 令 ， 得 到 Query OK 响应 确保 清除 过 程 顺利 地 进 
行 了 。 


mysql> FLUSH PRIVILEGES; 
Query OK, @ rows affected (09.10 sec) 


FLUSH TABLES 命 令 关 闭 当前 打开 或 使 用 的 所 有 的 表 ， 并 且 在 开始 
加 到 工作 之 前 ， 基 本 上 给 MySQL 服 务 喜 一 坚 秒 的 缓冲 时 间 。 当 绥 存 为 
守 ，MySQL 可 以 更 好 地 使 用 可 用 内 存 。 再 一 次 ， 我 们 得 到 Query OKI 


M. 


mysql> FLUSH TABLES; 
Query OK, @ rows affected {0.21 sec) 


FLUSH HOSTS 命 令 专 门 对 主机 缓存 表 起 作用 。 如 采 我 们 无 法 连接 
到 MySQL 服 务 器 ， 一 个 常见 的 原因 是 ， 对 一 个 特定 主机 的 连接 达到 了 
最 大 的 连接 数目 并 且 会 抛 出 错误 。 当 MySQL 看 到 很 多 的 连接 错误 ， 它 
i UE. FLUSH 
HOSTS 命 令 重 新 设置 这 一 过 程 并 且 再 次 允许 进行 连接 。 


mysql> FLUSH HOSTS; 
Query OK, @ rows affected (09.00 sec) 


FLUSH LOGS 命 令 天 闭 并 重新 打开 有 所 有 的 日 志文 件 。 如 末日 志文 件 
变 成 一 种 负担， 你 希望 开始 一 个 新 的 日 志文 件 ， 这 条 命令 创建 一 个 新 
的 、 空 的 日 志文 件 。 在 一 个 文件 中 亿 历 一 年 的 日 志 条 目 来 查找 错误 ， 可 
能 是 一 件 索 琐 的 事情 ， 因 此 ， 演 试 至 少 每 月 清空 日 志 。 


mysql> FLUSH LOGS; 
Query OK, ð rows affected (0.04 sec) 


要 了 解 有 关 FLUSH 的 更 多 信息 ， 请 参阅 位 于 
http://dev.mysql.com/doc/refman/5.5/en/flush. html 的 MySQL 手 册 。 


31.6 ”使 用 SHOW 命令 


SHOW 命令 有 几 种 不 同 的 用 法 ， 它 会 产生 输出 ， 显 示 关 于 MySQL 
数据 库 、 用 户 和 表 的 众多 有 用 信息 。 根 据 我 们 的 访问 级 别 ， 一 些 SHOW 
命令 将 不 能 供 我 们 使 用 ， 或 者 只 提供 少量 信息 。root 级 别 的 用 户 上 其 有 使 
用 所 有 SHOW 命令 的 能 力 ， 而 且 能 得 到 了 最 全 面 的 结 末 。SHOW 的 音 见 用 
法 如 下 ， 我 们 稍 后 会 详细 介绍 。 

SHOW GRANTS FOR user 

SHOW DATABASES [LIKE something] 

SHOW [OPEN] TABLES [FROM database name] [LIKE something] 

SHOW CREATE TABLE table name 

SHOW [FULL] COLUMNS FROM table name [FROM database name] [LIKE something] 
SHOW INDEX FROM table name [FROM database name] 

SHOW TABLE STATUS [FROM op name] [LIKE something] 


SHOW STATUS [LIKE something] 
SHOW VARIABLES [LIKE something] 


SHOW GRANTS fig S wR I -TAEHAE 2 FE ELI AMER 
eT SICA YE, JOE RR BT 1B 
户 权 限 的 请 求 的 时 候 。 通 过 SHOW GRANTS， 我 们 首先 检查 看 看 该 用 户 
是 不 是 已 经 拥有 了 所 请 求 的 权限 。 下 面 的 例子 查看 用 户 joeuser 的 可 用 权 
限 。 


SHOW GRANTS FOR joeuser@localhost; 


这 个 碍 询 的 结束 如 下 。 


| GRANT ALL PRIVILEGES ON *.* TO ‘joeuser'@' localhost’ IDENTIFIED | 
| BY PASSWORD ' *13883BDDBE566ECEFF@50@1CDE9B293303116521A' | 


1 rows in set (0.00 sec) 
如 果 我 们 不 是 root 级 别 用 户 或 joeuser 用 户 ， 将 会 得 到 一 个 错误 。 除 
非 我 们 是 root 级 别 用 户 ， 人 否则 ， 只 能 看 到 和 目 己 相关 的 信息 。 例 如 ， 
joeuser H HAUTE AA Kroon AAP a E 
SHOW GRANTS FOR root@localhost; 
1X ok BW BO BA I E o 
ERROR 1844: Access denied for user: 'joeuser@localhost' to database ‘mysql | 
FEAR ARNE, thE ORI Al. WRIA Erot Fl 
用 户 ， 不 可 以 使 用 其 中 一 些 命令 ， 或 者 使 用 它们 只 能 显示 有 限 的 信息 。 


还 有 一 些 其 他 常用 的 SHOW 命 令 ， 要 了 解 更 多 信息 ， 请 参阅 位 于 
http://dev.mysql.com/doc /refman/ 5.5/ en/ show. html 的 MySQL 手 册 。 


31.6.1 获取 有 关 数 据 库 和 表 的 信息 


在 本 书 前 面 ， 我 们 已 经 使 用 了 一 些 基 本 的 SHOW 命令 来 租 看 
MySQL 服 务 硕 上 的 数据 库 和 表 的 列表 。 复 习 一 下 ，SHOW DATABASE 
命令 怠 是 用 来 做 这 件 事情 的 ， 它 列 出 MySQL 服 务 右 上 的 所 有 数据 库 。 
如 下 是 结果 示例 。 


| Database | 
人 十 
| testDB | 
| mysql | 
a ee + 


2 rows in set (0.00 sec) 


当 我 们 选择 了 要 使 用 的 数据 库 后 ， 也 可 以 使 用 SHOW 列 出 数据 库 中 
的 表 。 这 个 例子 是 在 选择 testDB 数 据 库 之 后 ， 运 行 一 条 SHOW TABLES 
丛 询 的 结果 《你 列 出 的 表 可 能 有 上 所 不 同 ) 。 


ee + 
| Tables in testDB | 
HS areara N EAE EAS + 
| grocery inventory | 
| email | 
| master name | 
| myTest | 
| testTable | 
人 十 


5 rows in set (0.01 sec) 


如 果 为 SHOW TABLES 命 令 添 加 OPEN， 将 会 得 到 表 绥 存 中 的 所 有 
表 的 一 个 列表 ， 并 显示 它们 被 缓存 和 使 用 了 多 少 次 。 


SHOW OPEN TABLES; 


结 采 如 下 所 示 。 
2 et en ysl ea Ss ee 2 ee pein, aes a apes chek 6 etnies. be + 
| Database | Table | In use | Name locked | 
PAREAN srk ea ene < a Teus Ss Te + 
| mysql | procs priv | @ | @ | 
| mysql | db | @ | 8 | 
| mysql | host | g | @ | 
| testdb | grocery inventory | @ | @ | 
| mysql | user | @ | ð | 
| mysql | tables priv | @ | @ | 
| mysql | columns priv | @ | @ | 
+---------- +------------------- +-------- +------------- 十 


7 rows in set (0.00 sec) 


把 这 些 信 息 和 本 章 前 面 所 学 习 过 的 FLUSH TABLES 命 令 结合 起 来 使 
用 ， 将 能 够 帮助 我 们 保证 数据 库 更 顺利 地 运行 。 如 果 SHOW OPEN 
TABLES 显 示 表 绥 存 了 很 多 次 ， 但 是 当前 没有 使 用 ， 使 用 FLUSH 
TABLES 命 令 来 释放 内 存 。 


31.6.2 ”获取 表 结 构 信 息 


一 条 有 用 的 命令 是 SHOW CREATE TABLE， 它 所 做 的 事情 正如 其 
名 字 所 示 ， 它 显示 用 来 创建 指定 的 表 的 SQL 语 句 。 


SHOW CREATE TABLE grocery inventory; 


上 述 命 令 执 行 的 结果 如 下 。 


| grocery inventory | CREATE TABLE 'grocery_inventory' { 
ad int{11) NOT NULL auto_increment, 
‘item name’ varchar(5@} NOT NULL default ", 
‘item_desc' text, 
‘item price float NOT NULL default ‘@', 
‘curr qty' int{11) NOT NULL default ‘9', 
PRIMARY KEY {'id') 
} ENGINE=InnoDB DEFAULT CHARSET=latin1 
机 i ee i + 


1 row in set (@.08 sec) 
如 果 我 们 导出 表 结 构 ， 得 到 的 信息 和 这 个 基本 相同 ， 但 是 ， 如 果 我 


们 只 是 查找 一 个 特定 的 表 创 建 语 句 的 提示 或 简单 引用 的 话 ，SHOW 
CREATE TABLE 命 令 使 用 起 来 会 更 快 。 


如 果 你 需要 知道 表 的 结构 但 不 需要 SQL 命令 创建 它 ， 可 以 使 用 如 下 
的 SHOW COLUMNS 命 令 。 


SHOW COLUMNS FROM grocery inventory; 


得 询 的 结 来 如 下 。 


Piero. sue aoe as aes ett ose Ja et a ee re 下 省 十 
| Field | Type | Null | Key | Default | Extra | 
peceras geiau i fan ee ee eens f------ f----- f----+----- fe eee eee eee eee eee 十 
| id | int(11} | NO | PRI | | auto increment | 
| item_name | varchar(5@) | NO | | | | 
| item desc | text | YES | | | | 
| item price | float | NO | | | | 
| curr_qty | int{11) | NO | | | | 
有 ns aires as Temenan ae a Te dessi opeen + 


5 rows in set (0.01 sec} 


提示 : 





SHOW COLUMNS 命 令 和 DESCRIBE 命 令 彼 此 互 为 别名 ， 因 此 ， 它 们 做 同样 的 事情 。 


SHOW INDEX 命 令 显 示 了 一 个 特定 表 中 出 现 的 所 有 索引 的 信息 。 
BIA Fe 


SHOW INDEX FROM grocery_inventory; 


这 条 命令 产生 一 个 信息 完整 的 表 ， 从 列 名 到 索引 的 基数 。 表 31-1 描 
述 了 这 条 命令 所 返回 的 列 。 


表 31-1 SHOW INDEX 结 果 中 的 列 


Non_unique “| 1 或 0，1= 和 索引 可 以 重复 ，0= 索 引 不 可 以 重复 





Key_name 索引 的 名 字 


Seq_in_index | RIWI S; MIF AR 


Collation 列 的 排列 顺序 ， 要 么 是 A (升序 ) 要 么 是 NULL (不 排序 ) 


索引 中 唯一 值 的 数目 


ce renee 这 里 给 出 了 索引 字符 的 数 日 ， 如 果 整 个 键 都 


Null ERI EnaA NULLE 


产生 一 个 填 满 结果 的 、 广 泛 的 表 的 另 一 条 命令 是 SHOW TABLE 
STATUS 命令 。 这 条 命令 的 语法 如 下 所 示 。 





SHOW TABLE STATUS [FROM database name] LIKE ‘something’ 


这 条 命令 产生 一 个 填 满 了 信息 的 表 ， 其 范围 从 行 的 大 小 和 数目 到 
auto_increment 字 段 要 使 用 的 下 一 个 值 。 ences 文 条 命令 所 返回 的 


各 列 。 


表 31-2 SHOW TABLE STATUS 结果 中 的 列 


DÙ 
co 
R 


Di HH 


表 的 名 子 


这 个 表 所 使 用 的 存储 引擎 


表 的 *.frm 文 件 的 版 本 


Version 


行 的 存储 格式 ， 固定 的 、 动 态 的 或 压缩 的 


Row_ format 





47 BL 
Avg_row_length | 行 的 平均 长 度 
Data_length 数据 文件 的 长 度 


Max_data_length | 数据 文件 的 最 大 长 度 


Index_length 索引 文件 的 长 度 


Data free 分 配 了 但 没有 使 用 的 字 节 数 


Auto increment | 一 个 auto_increment 字 段 中 使 用 的 下 一 个 值 





pe tT] Z 

Z f = 

四 5 D 
D 


Create time 创建 表 的 日 期 和 时 间 《〈 以 日 期 时 间 格 式 ) 


Ca [anane Ce 
tine | aaaanmemn ce 
extn fear 


表 的 校 验 和 值 ， 如 果 使 用 校 验 和 的 话 
用 于 CREATE TABLE 语 名 的 任何 附加 选项 


EREE 创建 表 的 时 候 所 添加 的 任何 说 明 。 此 外 ，InnoDB 表 使 用 这 个 列 来 
报告 表 空间 中 的 空 自 空间 


31.6.3 ”获取 系统 状态 





SHOW STATUS 和 SHOW VARIABLES 命 令 快 速 提 供 有 关 数 握 库 服 
务 器 的 重要 信息 。 这 些 命令 的 语法 就 是 SHOW STATUS 和 SHOW 
VARIABLES. 


有 超过 300 个 状态 变量 作为 SHOW STATUS 的 输出 ， 但 最 有 用 的 如 
下 所 示 。 


e Aborted_connects 壬 试 连接 到 MySQL 的 失败 次 数 。 任 何 时 
做 ， 我 们 看 到 一 个 放弃 的 连接 ， 束 应 该 调 伍 这 个 问题 。 可 能 和 脚本 
中 错误 的 用 户 名 和 知人 码 有 天 ， 或 者 所 允许 的 同时 连接 的 数目 设置 得 





太 低 ， 为 了 防止 站 点 流量 过 大 。 
在 当前 正常 运行 时 期 ， 壬 试 连接 到 MySQL 服 务 





e Connections 





的 连接 的 总 数 。 
e Max used connections 在 当前 正常 运行 时 期 ， 同 时 使 用 的 连接 
的 最 大 数目 。 





超过 long_query_time 有 时间 的 查询 的 总 数 ， 
long_query_time 默 认为 10 秒 。 如 采 较 慢 的 得 询 多 于 一 个 ， 应 该 研究 
一 下 SQL 语法 。 

在 当前 正常 运行 时 ， 服 务 右 已 经 局 动 的 恕 秒 数 。 


e Slow_queries 





e Uptime 


EME F http://dev.mysql.com/doc/refman/5.5/en/show-status.html 的 
MySQL 手 册 中 ， 可 以 找到 SHOW STATUS 变量 的 一 个 完整 列表 ， 及 其 
参数 的 说 明 。 


SHOW VARIABLES 命 令 产 生 325 个 结果 ， 这 些 结果 会 控制 MySQL 
的 一 般 运 行 ， 并 且 包 括 如 下 项 目 。 


显示 MVySQL 服 务 需 在 一 个 连接 答 试 放弃 之 前 





等 待 的 秒 数 。 
e max_connections 
间 时 连接 的 数目 。 
MySQL 运 行 的 端口 。 
MySQL KRH, 
MySQL 的 版 本 与 。 





在 一 个 连接 被 拒绝 之 前 ， 所 允许 到 MySQL 的 





e port 





e table_type 


e version 





TEM. ¥ http://dev.mysql.com/doc/refman/5.5/en/show-variables.html 的 
MySQL 手 册 可 以 看 到 SHOW VARIABLES 所 返回 的 变量 的 一 个 完整 列 


表 ， 以 及 它们 的 值 的 说 明 。 妆 你 知道 了 所 拥有 的 值 ， 就 可 以 在 MySQL 
配置 文件 或 局 动 命令 中 修改 它们 。 


Ailey Ne 


运行 一 个 优化 的 MySQL 服 务 磺 ， 首 先 从 所 使 用 的 人 硬件 和 操作 系统 
开始 。 系 统 的 CPU 应 该 足够 快 ， 并 且 应 该 有 足够 的 RAM 用 来 弥补 CPU 的 
不 足 。 如 果 MySQL 和 其 他 的 进程 〈 如 Web 服 务 器 ) 分 享 资 源 ， 尤 其 应 该 
如 此 。 


此 外 ， 所 使 用 的 硬盘 驱动 器 也 很 重要 ， 因 为 小 的 硬盘 驱动 器 限制 了 
可 以 在 数据 库 中 存储 的 信息 的 数量 。 硬 盘 驱 动 器 的 寻 道 时 间 也 很 重要 ， 
慢 的 寻 道 速度 会 导致 服务 器 性 能 全 面 降低 。 操 作 系统 应 该 不 占用 太 多 机 
复 资 源 ， 并 且 应 该 和 MySQL 共 再 资源 而 不 是 目 己 使 用 所 有 的 资源 。 


MYVySQL 的 一 些 关 键 月 动 参数 包括 key_buffer_size 和 table_cache， 还 
有 其 他 一 些 参数 。 可 以 在 示例 MySQL 配 置 文件 中 找到 基准 值 ， 或 者 可 
以 修改 这 些 变 量 并 但 看 服务 右 性 能 ， 看 看 我 们 古人 否 找 到 了 适合 坏 境 的 正 
确 参 数值 。 


除了 便 件 和 软件 优化 ， 还 有 表 的 优化 ， 以 及 SELECT 但 询 的 优化 。 
表 的 优化 使 用 OPTIMIZE 命 令 ， 我 们 可 以 重新 使 用 未 使 用 的 空间 。 我 们 
可 以 通过 使 用 EXPLAIN 命 令 看 看 售 询 优化 有 多 么 好 《或 者 多 么 不 
好 ) 。 最 终 的 输出 将 会 显示 是 人 否 以 及 何 时 使 用 了 有 索引， 以 及 是 耕 可 以 使 
用 索引 来 提高 给 定 碍 询 的 速度 。 


对 MySQL 服 务 器 多 加 注意 ， 以 确保 它 持 续 顺 利 地 运行 。 像 FLUSH 
和 SHOW 这 样 的 基本 的 管理 命令 ， 可 以 帮助 我 们 识别 和 快速 修复 潜在 的 
问题 。 所 有 这 些 命令 都 设计 给 MySQL1 毫 秒 的 嘴 息 时 间 ， 如 果 它 处 在 一 
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以 及 系统 是 如 何 运 行 的 结构 性 信息 。 


31.8 Q&A 

Q: 在 单个 的 服务 中 ，MySQL 可 以 利用 多 CPU 的 优势 吗 ? 
A: 绝对 可 以 。 如 果 操 作 系 统 文 持 多 CPU，MYySQL 将 会 利用 其 优 

点 。 然 而 ， 根 据 操 作 系 统 不 同 ， 使 用 多 CPU 的 MySQL 的 调 校 有 上 所 不 

Q: 要 使 用 OPTIMIZE 命 令 必须 有 什么 级 别 的 许可 ? 


A: 对 一 个 表 具 有 INSERT 权 限 的 任何 用 户 都 可 以 执行 OPTIMIZE 命 
令 。 如 果 一 个 用 户 只 有 SELECT 许可 ， 则 不 能 执行 OPTIMIZE 命 令 。 


31.9 RAS 


这 个 实践 练习 设计 用 来 帮助 你 预测 可 能 的 问题 、 复 习 已 经 学 过 的 知 
识 ， 并 且 开 始 把 知识 用 于 实践 。 


， 哪个 MySQL K BUFR TT BE is BIBT PS RIA TW HA 
cea 


a 
R 
i 


， 哪 条 SQL 命令 清除 表 的 结构 ? 
， 哪 条 FLUSH 命 令 重 置 MySQL 日 志文 件 ? 


4. 要 快速 确定 MySQL 十 合 文 持 InnoDB 表 ， 应 该 使 用 SHOW 
STATUS 还 是 SHOW VARIABLES? 


5. 编写 一 条 SQL 语句 ， 使 得 我 们 能 够 看 到 用 来 创建 myTable 表 的 
SQL 语句 。 


oy 


— 


N 


benchmark() PK 2 


. OPTIMIZE 


FLUSH LOGS 
SHOW VARIABLES 


SHOW CREATE TABLE myTable 


EA wel 


1. WRR IRS as A roota ABR, 1BrXkey_buffer_size fil 
table_cachel{H, JF HEREA Jata{rbenchmark() RIAL, AA AIT 
YY TA) EY X al « 


2. 对 数据 库 中 所 创建 的 所 有 表 执 行 OPTIMIZE 命 令 以 清除 任何 结构 


问题 。 


3. 使 用 SHOW STATUS 命 令 来 获取 有 关 MySQL 服 务 器 的 信息 ， 然 
后 执行 FLUSH 命 令 清 除 服 务 器 。 在 每 条 命令 之 后 ， 再 次 使 用 SHOW 
STATUS 来 看 看 哪 条 命令 影响 到 SHOW STATUS 结果 显示 中 的 哪些 结 
R, 


第 32 章 ”软件 升级 


ERE, RITKA J: 


。 LAH ER Er ENT HI ERIE ETE o 

。 如 何在 MySQL 的 次 版 本 之 间 升 级 。 
。 如 何在 Apache 的 次 版 本 之 间 升 级 。 
。 如 何在 PHP 的 次 版 本 之 间 升 级 。 


整个 本 书 中 ， 我 们 已 经 提醒 过 你 得 找 有 关 升 级 PHP、Apache 和 
MySQL 的 新 版 本 的 信息 ， 并 且 要 留意 更 新 。 此 外 ， 还 介绍 了 如 何在 构 
建 的 时 候 同 PHP 汐 加 蕊 能 ， 但 是 这 只 是 在 安 靖 软件 的 时 候 。 在 这 个 简短 
的 一 草 中 ， 我 们 将 学 习 在 经 历 了 一 段 正 章 运 行 时 间 之 后 如 何 更 新 已 经 安 
浅 过 的 软件 ， 而 不 会 给 系统 市 来 严重 破坏 。 


32.1 {= PATE eA F 


你 应 该 已 经 收藏 了 Apache、PHP 和 MYySQL 的 站 点 。 你 曾经 使 用 过 这 
些 技术 6 天 还 是 6 年 都 无 关 紧 要 ， 经 名 回头 看 看 这 些 站 点 总 是 有 必要 的 
(我 总 是 这 么 做 ) 。 如 果 访 问 Web 站 点 的 主要 原因 是 获取 有 关 更 新 的 信 
轧 ， 你 可 以 订阅 一 个 目 愿 订阅 的 邮件 列表 。 


。 要 了 解 MySQL 信 息 ， 到 http:VWlists.mysql.comy 并 订阅 MySQL 信 息 列 
HK o 

。 要 了 解 Apache 信 息 ， 
到 http:/www.apache.org/foundation/mailinglists.html #71] |] Apache% 
闻 和 信息 列表 。 

e 2 J fAPHP{a i, #http://www.php.net/mailing.lists.php JF] [J PHP 
信息 列表 。 


何 时 升级 


正如 安 骤 章节 所 所 到 的 ， 任 何 时 候 ， 当 开 人 及 着 及 现 有 必要 的 时 候 吏 
会 友 布 钦 版 本 变化 的 狐 软 件 ， 而 没有 任何 特定 时 间 表 。 但 是 ， 仪 仅 次 版 
本 及 生 了 变化 ， 并 不 一 定 意 味 痢 你 应 该 马上 跟 进 并 升级 软件 。 然 而 ， 有 
时 候 还 是 应 访 升 级 。 


软件 发 布 明 党 这 守 major.minor.revision 的 格式 ， 例 如 PHP 5.4.0 的 主 
版 本 是 5， 次 版 本 是 4， 修 订 编 号 是 0。 也 有 可 能 是 这 样 ， 实 际 上 修订 中 
所 做 的 修改 可 能 “很 小 ”， 但 是 发 布 还 是 还 是 将 其 当做 一 次 修订 。 


当 公 布 一 个 安全 性 修正 的 时 候 ， 应 该 立即 升级 软件 。 通 币 ， 安 全 性 
Pe) el PR BR I ee ae a AL, AREENA, (AA IN tee Al 
FI MAE | AP A A © EEK NEE 
Zia, VRAD E OAM A i 2 RANE RTE ea, FFE 
很 快 将 会 看 到 一 个 升级 的 发布 。 这 时 候 ， 你 应 该 立即 升级 ， 即 便 你 没有 
使 用 引 友 安全 问题 的 特定 元 素 。 漏 铜 融 是 漏洞 ， 为 什么 让 它 隐 藏 呢 ? 


下 面 是 Apache 更 新 日 志 的 一 个 例子 ， 它 记录 了 版 本 2.0.61 和 2.0.63 
之 则 发 生 的 变化 (版 本 2.0.62 没 有 公开 发 布 ) ， 这 是 升级 所 需要 的 一 个 
指 路 牌 。 


SECURITY: CVE-2007 -6388 {cve.mitre.org)} 
mod status: Ensure refresh parameter is numeric to prevent a possible XS9 
attack caused by redirecting to other URLs. 


首要 的 原则 是 ， 如 果 单 词 “ 安 全 性 (security) ”出 现在 更 新 日 志 的 任 
何 位 置 ， 那 么 应 该 并 即 升 级 。 在 PHP5 EJ HAP MEF 
http://www.php.net/ChangeLog-5.php ) ， 将 与 安全 相关 的 修改 收集 到 一 
起 ， 并 且 由 于 其 重要 性 而 放 在 这 个 列表 的 顶部 。 


然而 ， 如 果 只 是 一 个 维护 友 布 ， 意 味 看 它 包 含 了 在 正常 的 开发 中 友 
生 的 错误 修复 和 一 般 性 维护 ， 你 可 能 不 十 要 立即 丢 莽 原 有 内 容 来 升级 软 
件 。 下 面 是 Apache 和 PHP 喝 新 日 志 中 维护 性 项 目的 一 些 例子 。 


mpm_winnt: Eliminate wait for many objects. Allows the clean shutdown of the 
server when the MaxClients is higher then 257, in a more responsive manner. 
Fixed bug #43137 ({rmdir(} and rename({) do not clear statcache). 


如 采 更 新 列表 中 没有 和 你 、 你 的 工作 以 及 你 的 环境 相关 的 东西 ， 那 
么 ， 你 可 以 推迟 更 新 ， 百 到 日 程 安排 中 的 休 奶 日 或 者 一 个 雨天 。 例 如， 
如 果 在 PHP 的 一 个 维护 友 布 中 修复 的 所 有 bug 部 是 和 Windows 平 台 相 关 


I, We FELinux Fie{TPHP, MAMA DAA I, TC 
Woe 
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新 的 特性 或 得 到 真正 和 我 们 的 工作 或 环境 相关 的 漏洞 修复 。 


32.2 升级 MySQL 


无 论 你 是 使 用 UNIX/Linux， 还 是 使 用 Windows， 升 级 MySQL 的 次 
版 本 都 很 简单 ， 只 要 安 闭 新 的 版 本 了 驶 好 像 其 他 版 本 不 存在 一 样 。 


提示 : 








在 升级 到 MySQL 的 一 个 新 的 次 版 本 之 前 ， 备 份 好 已 有 的 数据 库 。 


在 同样 的 基本 版 本 中 更 新 MySQL 很 容易 ，5.4 是 基本 版 本 ，5.5 是 一 
个 基本 版 本 ， 以 此 类 推 。 这 意味 看 在 5.5.x 组 内 的 任何 次 版 本 升级 束 像 在 
旧 软 件 之 上 安装 新 软件 一 样 的 和 价 单 。 通 过 Windows Installer， 你 可 以 看 
到 这 一 切 。 从 二 进 制 发 布 安装 的 UNIX/Linux 用 户 ， 只 需要 把 “mvysql” 符 
写 目 录 重 新 链接 到 新 的 、 未 解 包 的 有 友 布 ， 以 此 作为 安 猴 进程 的 一 部 分 。 


如 琳 在 升级 过 程 中 直到 问题 ， 参 考 位 于 
http://dev.mysql.com/doc/refman/5.5/en/upgrading. html 的 故障 排除 说 明 。 
然而 ， 升 级 MySQL 的 次 版 本 对 我 来 说 总 是 顺利 的 过 程 ， 在 哪个 平 侣 上 
HBE AIE o 
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数据 的 校对 类 型 。 


# myisamchk -r -q --set-collation=new collation name 


32.3 Ft% Apache 


AIMySQL—#t, FFR E E HT Aa EA pache Fil E — x ZEAE FEM E] 
样 的 步骤 。Windows 用 户 得 益 于 Installer 恬 用 程序 ， 它 会 目 动 检测 之 前 的 
版 本 ， 删 除 核 心 部 件 并 安装 新 的 。 然 而 ，Windows Installer 将 会 你 留 已 
有 的 配置 文件 。 我 们 还 要 人 负 贡 升级 任何 其 他 的 特定 于 版 本 的 模块 ， 例 如 
mod_ssl， 访 模块 是 绑 定 到 Apache 的 特定 版 本 的 。 


对 于 UNIX/Linux 用 户 ， 这 个 过 程 遵照 和 最 初 安装 同样 的 步骤 。 当 我 
们 解压 缩 新 的 发布 包 ， 它 创建 一 个 以 新 厂 本 号 命名 的 目录 。 例 如 ， 如 末 
我 们 之 前 的 版 本 写 古 2.2.17， 并 且 要 升级 到 2.4.1， 对 应 的 目录 名 将 会 分 
别 是 httpd-2.2.17 和 httpd-2.4.1。 


Apache 的 实际 安 贸 目录 由 你 确定 ， 当 你 运行 configure 脚 本 的 时 候 确 
定 ， 如 下 和 面 的 例子 所 示 。 


# ./configure --prefix=/usr/local/apache2 


在 运行 了 configure 脚 本 生成 了 Apache 的 新 版 本 之 后 ， 只 要 像 第 一 次 
安装 Apache 那 样 经 历 make 和 和 make install 过 程 。 


MÆ, MIZER EIH Apache% EARI Apache kÆ í, 
甚 全 可 以 在 旧 的 httpd 二 进 制 仍 和 在 运行 的 时 候 这 么 做 。 只 要 确保 备份 了 配 
置 文件 以 防 出 错 。 然 而 ， 如 采 你 更 习惯 在 不 同 的 目录 中 安 猴 新 版 本 ， 这 
也 不 错 ， 只 十 必须 要 把 所 有 和 Web 相 天 的 文件 〈 即 文档 根 目 录 下 的 所 有 
内 容 ) 移动 到 新 目录 下 ， 并 且 让 所 有 相应 的 编辑 都 反映 到 新 的 httpd.conf 
文件 。 选 择 什么 方法 取决 于 你 ， 一 种 方法 需要 较 多 的 文件 移动 ， 男 一 种 


方法 需要 较 多 的 配置 。 


在 UNIX/Linux 上 升级 了 Apache 之 后 ， 应 该 也 重新 编译 PHP 模 块 。 
Windows 用 户 不 需要 有 一 个 重新 编译 的 模块 ， 但 是 应 该 确保 相应 的 与 
PHP 相 关 的 更 新 也 出 现在 httpd.conf 文 件 中 ， 这 关系 到 驻 留 在 PHP 目 录 树 
下 的 模块 的 加 载 。 


修改 Apache 而 不 需要 升级 


假设 我 们 需要 从 Apache 玖 加 或 删除 功能 而 不 需要 升级 到 一 个 新 的 次 
版 本 。 一 个 例子 就 是 ， 添 加 一 个 新 的 模块 ， 或 者 把 系统 上 使 用 的 
OpenSSL 升 级 到 新 的 版 本 。 


在 这 种 情况 下 ，UNIX/Linux 用 户 应 访 找 到 己 有 的 谣 目 录 《〈 例 如 
httpd-2.2.17) ， 并 且 在 命令 行 输入 make clean 。 基 本 上 ， 这 将 重 置 
makefiles 以 便 我 们 可 以 重新 编译 Apache 而 不 需要 依赖 以 前 的 、 绥 存 的 
值 。 在 make clean 命 令 之 后 ， 用 新 的 参数 运行 configure 脚 本 ， 并 用 再 次 
进行 make 和 make install 过 程 。 在 这 种 情况 下 ， 应 该 不 需要 重新 编 详 PHP 
模块 。 


通过 在 httpd.conf 中 对 相应 的 行 去 挥 注释 ， 或 者 如 果 这 些 行 已 经 不 存 
在 就 添加 那些 行 ，Windows 用 户 能 够 激活 预 编译 模块 。 


32.4 升级 PHP 


既然 UNIX/Linux 用 户 可 以 通过 各 种 编 详 选项 来 涩 加 如 此 多 的 功能 到 
PHP， 你 升级 或 修改 PHP 可 能 比 Apache 和 MVySQL 要 频繁 很 多 。 不 管 你 是 
升级 到 一 个 新 的 次 版 本， 还 是 傈 单 地 琴 加 新 功能 ， 或 者 删除 条 些 不 再 需 
要 的 功能 ， 修 改 已 有 版 本 的 过 程 确 实 和 第 一 次 安 竣 它 的 过 程 相同 : 
configure、make 和 make install. make install 步 又 把 PHP 模 块 放 入 到 
Apache 目 录 树 的 相应 位 置 。 当 新 的 模块 放 入 到 指定 位 置 ， 重 新 局 动 
Apache， 新 版 本 的 PHP 就 开始 使 用 了 。 


如 末 你 升级 到 PHP 的 一 个 新 的 次 厂 本 ， 当 我 们 提取 了 奴 布 文 档 ， 将 
会 生成 一 个 根据 版 本 号 的 完全 不 同 的 目录 树 。 在 新 的 目录 结构 中 执行 
configure、make 和 make install 步 又 ， 并 且 将 会 生成 一 个 新 的 PHP 模 块 ， 
它 独 立 于 其 他 的 模块 。 


Windows 用 户 有 一 组 不 同 的 任务 要 执行 ， 同 一 个 已 有 的 模块 瀛 加 新 
功能 只 需要 通过 在 php.ini 中 去 挥 其 条 目的 注释 并 重 狐 局 动 Apache， 束 可 
以 激活 该 模块 。 升 级 到 一 个 新 的 次 版 本 则 需要 下 载 一 个 新 的 用 布 文件 。 
这 个 文件 的 内 容 随后 提取 到 一 个 按照 它 所 表示 的 版 本 而 命名 的 目录 中 。 
然后 ， 必 须 按照 安 儿 所 需 的 步 又， 相应 地 配置 php.ini， 因 为 每 个 厂 本 生 
成 一 个 不 同 的 文件 。 最 后 ， 在 Apache 的 httpd.conf 文 件 中 修改 和 PHP 相 关 
的 路 径 名 ， 并 且 重 新 局 动 服务 左 ， 新 的 PHP 版 本 了 台 可 以 使 用 了 。 


使 用 PECL 和 PEAR 扩 展 PHP 


可 以 从 PECL (PHP Extension community library, PHPH ESHE, 


位 于 http:/pecl.php. net/ ) 和 PEAR (PHP Extension and application 
repository，PHP 扩 展 与 应 用 库 ， 位 于 http:// pear.php.net/ ) 获取 用 户 创建 
的 扩展 和 应 用 程序 。 这 些 站 点 都 有 规则 和 样式 管理 ， 因 此 ， 从 这 里 下 载 
的 任何 内 容 都 共有 很 高 的 质量 


如 果 你 在 查找 PHP 安 装 的 其 他 扩展 ， 请 但 看 PECL。 如 果 你 在 查找 
集成 到 应 用 程序 中 的 开源 代 公 的 库 ， 请 查看 PEAR。 


32.5 ”小 结 


这 个 简短 的 一 章 提 供 了 升级 MySQL、Apache 和 PHP 当 前 版 本 安装 的 
一 些 指 导 。 我 们 学 习 了 如 何 找到 更 新 ， 以 及 如 何 评估 升级 到 新 版 本 的 重 
要 性 。 此 外 ， 还 学 习 了 升级 或 修改 MySQL、Apache 和 PHP 的 步骤 。 


32.6 ”实践 练习 


这 个 实践 练习 设计 用 来 帮助 你 预测 可 能 的 问题 、 复 习 已 经 学 过 的 知 
识 ， 并 且 开 始 把 知识 用 于 实践 。 


e E 

1. TERENI, KWEN, EFAA, RANZA 
表示 它 。 和 

2. 升级 到 任何 软件 的 一 个 新 的 次 版 本 所 考 夸 的 首要 原因 是 什么 ? 


3. 什么 命令 将 清空 之 前 的 makefiles 和 缓存 设 置 ? 


oy 


— 


N 


完整 的 版 本 号 应 该 是 3.4.14。 
开 必 者 发 现 或 修复 了 安全 性 问题 。 


make clean 命 令 。 


EA wel 


在 阅读 到 本 章 的 时 候 ， 可 能 已 经 有 了 与 本 书 CD 所 包含 的 或 者 你 在 
本 书 开始 所 下 载 安 装 的 PHP、MySQL 和 Apache 的 不 同 版 本 了 。 因 此 ， 如 
果 你 能 够 升级 〈 也 就 是 说 ， 如 果 你 没有 使 用 虚拟 主机 环境 的 话 ) ， 选 择 
一 种 或 多 种 方法 ， 并 完成 升级 过 程 。 


第 33 草 ”使 用 应 用 程序 框 染 


在 这 个 简单 的 一 半 中 ， 我 们 将 学 习 如 下 内 容 : 


。 应 用 程序 框架 是 什么 ， 它 们 有 什么 用 。 
o 软件 架构 的 模型 -视图 -控制 器 模式 的 相关 基础 知识 。 
。 考虑 和 和 安 泪 一 些 流行 的 PHP 应 用 程序 框 染 。 


本 书 已 经 教授 了 在 一 个 Web 站 扣 中 创建 动态 功能 的 基础 知识 ， 既 包 
括 使 用 单个 的 脚本 来 增强 功能 或 显示 动态 数据 ， 也 包括 将 一 系列 的 脚本 
以 东 种 一 致 的 形式 结合 在 一 起 ， 以 生成 基于 Web 的 应 用 程序 。 当 你 试图 
开始 一 个 较 大 的 项 目的 时 候 ， 你 很 可 能 想 要 使 用 其 他 人 很 可 能 已 经 在 目 
己 的 项 目 中 用 到 过 的 功能 ， 而 你 没有 时 间 也 不 愿意 曾 新 友 明 轮子 ， 这 时 
各， 应 用 程序 框 染 会 成 为 你 的 新 的 好 朋友 。 


33.1 理解 应 用 程序 框架 


究 其 本 质 ， 应 用 程序 框架 就 是 一 组 库 和 模板 ， 它 们 允许 你 快速 地 开 
友 功 能 丰富 的 动态 站 点 和 Web 应 用 程 计 ， 而 不 十 要 从 头 开始 构建 每 一 个 
模块 。 你 可 能 还 记得 ， 在 本 书 第 五 部 分 中 ， 我 忌 古 很 仔细 地 提 人 a 到， 给 出 
的 示例 只 是 创建 脚本 实现 总 体 目 标的 众多 方法 中 的 一 种 。 使 用 应 用 程序 
框 染 ， 使 得 你 可 以 说 “我 知 记 有 很 多 种 方法 来 创建 一 个 登录 过 程 ( 或 购 
物 车 、 论 坛 等 等 ) ， 并 且 ， 这 些 方 法 不 是 从 头 开 始 ， 可 以 用 应 用 程序 杠 
TRIN TIE RE.” 


除了 针对 第 用 功能 香 用 一 个 稳定 的 代码 这 一 明显 的 好 处 ， 使 用 框 淋 
还 可 以 帮助 开 友 者 你 持 一 人 怪 的 软件 染 构 模式 。 在 PHP 框 染 的 例子 中 ， 这 
种 模式 通常 是 模型 -视图 -控制 器 (MVC) 模式 ， 我 们 将 在 下 一 小 节 中 更 
详细 地 介绍 它 。 框 架 以 及 由 此 引 妥 的 菏 些 软件 架构 所 具有 的 万 一 个 方面 
优点 是 ， 能 够 为 应 用 程序 实现 一 种 稳定 且 一 致 的 三 层 染 构 。 在 一 个 三 层 
染 构 中 ， 我 们 有 一 个 表示 客户 响 的 物理 层 ， 以 及 一 个 在 你 的 控制 管理 下 
的 应 用 程序 和 数据 库 。 换 名 话说， 用 户 的 Web 浏 览 耸 或 者 移动 设备 〈 竹 
Pin) 直接 连接 到 应 用 程序 〈 通 向 是 应 用 程序 的 表现 层 ， 或 痢 说 你 在 浏 
贷 秦 中 所 看 到 的 内 容 ) ， 并 进而 进入 数据 库 以 获取 要 显示 的 数据 。 


了 解 一 些 凡 容 管 理 系统 征 很 重要 的 ， 它 们 都 是 可 以 目 行 安 放 的 软件 
包 ， 例 如 WordPress (http://www.wordpress.org )、Drupal 
(http://www.drupal.org ) 和 Joomla (http://www.joomla.org )。 从 技术 上 讲 这 
EEA ELAS AN ee VEER, (Eze, FY DA eco A ce HE TI DA BE VK A 
己 喜 欢 的 风格 ， 从 而 重用 它们 的 功能 丰富 的 代码 。 


33.2 ”使 用 MVC 模 式 


任何 软件 染 构 模式 的 目标 之 一 是 ， 提 供 一 致 的 结构 、 元 素 以 及 特性 
和 属性 等 ， 以 确保 应 用 程序 内 部 的 工作 对 于 那些 开发 者 和 维护 者 来 说 是 
透明 的 。 考 谍 一 下 这 种 情况 ， 一 个 人 独 目 编 与 了 一 于 应 用 程序 ， 而 没有 
这 从 任何 结构 性 的 借 式 。 当 一 个 新 人 加 入 团队 ， 或 者 最 初 的 编写 者 离开 
的 时 候 ， 将 会 友 生 什么 事情 ? 没有 一 个 基础 的 和 共 圣 的 知识 体系 ， 例 
ON, WEA ELFEN, ABA A a te BK Be IN TA] BERA 
ASTOR, SEAN BES ARETE MAR ES LE JARA ET le 
到 的 问题 了 。 


MVC 软 件 染 构 模 式 是 为 基于 Web 的 应 用 程序 而 准备 的 ， 并 且 实 际 
上 ， 很 多 应 用 程序 (或 者 甚至 只 是 动态 站 点 ) 都 得 从 这 一 模式 的 茶 个 厂 
本 ， 而 这 么 做 坚 个 中 力 。 图 33-1 以 一 种 基本 的 方式 说 明了 MVC 的 概念 ， 
其 中 的 实 线 表示 三 个 组 件 中 的 每 两 个 之 间 的 直接 关联 ， 而 虚线 表示 作 些 
部 分 之 间 有 时 候 会 有 间接 的 关联 。 





图 33-1 MVC 模 式 的 图 形 表示 
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型 一 一 存储 并 分 隔 数 据 。 它 本 里 不 是 一 个 数据 库 ， 但 是 ， 可 以 
ne EREE PE 昌 的 部 分 ， 然 
后 ， 它 会 将 提取 的 信息 显示 给 用 请。 
rai 这 束 像 你 的 这 样 一 部 分 脚 
本 ， 它 们 确定 个 做 任何 其 他 事情 ， 只 是 显示 你 从 数据 库 获 取 的 数 
扼 。 
fe hl] a 一 一 系统 中 的 一 个 操作 和 梓 调 用 的 方式 。 通 币 ， 我 们 称 之 为 
应 用 程序 的 “大 脑 *"， 它 确定 如 何 使 用 模型 来 得 到 操作 的 一 个 成 功 的 
ZK o 


® 











还 有 其 他 的 一 些 方法 来 思考 一 个 MVC 应 用 程序 的 流程 。 假 设 网 上 
商店 的 一 位 用 户 正在 浏览 商品 页 面 ， 并 且 想 要 将 商品 添加 到 她 的 购物 车 
中 ， 


1. 用 户 使 用 鼠标 按 下 页 面 上 的 一 个 提交 按钮 。 
2. 控制 蕉 接收 表单 操作 并 且 将 信息 及 送 给 模型 。 


3. 为 了 辐 用 户 显 示 一 个 换 作 并 更 改 其 已 变 化 的 数据 ， 视 图 和 模型 
交流 数据 ， 以 将 数据 取出 并 显示 给 用 户 。 


4. 控制 邢 等 竺 更 多 的 用 户 操作 ， 然 后 ， 这 个 循环 将 重复 。 


本 草 中 所 介绍 的 每 个 PHP 框 染 部 使 你 可 以 很 容易 地 将 MVC 棕 式 应 用 
到 你 的 软件 应 用 程序 。 很 多 其 他 的 PHP 框 架 也 这 么 做 ， 尽 管 你 可 能 不 会 
选择 该 模式 ， 但 还 是 推荐 你 这 么 做 ， 以 使 得 应 用 程序 后 续 的 测试 、 开 


有 发、 部署 和 维护 更 容易 。 


要 了 解 天 于 MVC 模 式 的 更 多 示例 和 说 明 ， 请 参见 Jeff Atwood 的 博 
ae X “Understanding Model-View-Controller”， 位 于 
http://www.codinghorror.com/ blog/2008/05/understanding-model- view- 


controller.html 。 


33.3 ”安装 和 使 用 PHP 应 用 程序 框架 


在 编写 本 书 的 时 候 ， 全 球 的 开发 者 使 用 超过 30 种 的 PHP 应 用 程序 框 
架 。 即 使 选择 常用 的 一 些 来 介绍 的 话 ， 工 作 量 也 不 小 。 我 在 本 章 中 挑选 
的 这 些 有 痢 《〈 相 对 ) 较 长 的 历史 ， 并 且 其 开发 者 社区 有 看 上 升 的 趋势 。 
实际 上 ， 在 你 评估 一 计 框 架 是 否 适 用 于 上 自己 的 用 途 的 时 候 ， 有 3 个 和 代 
人 码 不 相关 的 方面 需要 考 不 : 它 是 否 有 一 段 时 间 了 ， 以 及 是 否 稳 定 ? 人 们 
积极 地 使 用 它 吗 ? 开发 它 的 公司 或 开发 者 组 织 积 极地 维护 它 吗 ? 


选择 应 用 程序 框 染 的 时 候 的 其 他 考虑 还 包括 如 下 几 个 方面 。 


。 人 确定 框 染 是 合 最 适合 于 你 要 开 友 的 应 用 程序 的 类 型 ， EER RG 
合 于 电子 商务 ， 一 些 框架 很 适合 于 内 容 友 布 ， 有 些 框架 则 两 者 都 适 
人 
O o 


WAKE ANE 2 BARGE S DE MEH ER APR, MRE 
PEAT, “ele te AE E H AIRAN o 

WATE AMER eB is Ah PHP PRR IRS feo UNE, TEETER 
无 法 控制 自己 的 服务 占 ， 并 且 由 此 无 法 修改 安 六 的 库 和 模块 的 话 ， 
你 将 无 法 使 用 该 框 染 。 


接 下 来 的 几 个 小 节 介 绍 3 种 流行 的 开源 PHP 应 用 程序 框 染 ，Zend、 
CakePHP 和 Codelgniter。 再 次 强调 一 下 ， 实 际 上 不 止 这 3 个 框 染 可 用 ; 
Symfony (http://www.symfony.com/ )#1l Yii(http:// www.yiiframework.com/ 
) 也 经 常 进入 “最 佳 候 选 ” 的 清单 。Wikipedia 维 护 了 PHP 应 用 程序 框架 的 
一 个 全 面 的 列表 ， 位 于 


http://en.wikipedia.org/wiki/Comparison_of_web_application_frameworks#P) 


33.3.1 Zend Framework 


如 膝 你 要 开 友 企业 级 的 PHP 应 用 程序 的 话 ， 应 该 认真 地 考虑 从 Zend 
Framework 开 始 (http://framework.zend.com/ )。 你 可 能 听 说 过 Zend 公 司 ， 
也 束 古 开 友 Zend Framework 的 公司 ， 其 创始 人 从 PHP 语 言 诞 生 的 时 候 吏 
是 该 语言 的 贡献 者 之 一 。 该 框架 核心 的 PHP 引 擎 常常 称 作 Zend 引 擎 。 换 
句 话说， 如 条 按照 我 前 面 提 到 的 那些 标准 来 评估 Zend Framework 的 话 ， 
很 难 找到 一 个 比 它 更 加 稳定 、 时 间 更 长 或 者 说 更 多 人 积极 参与 开发 的 框 


Te So 


尽管 该 软件 的 名 字 叫 做 Zend Framework， 并 且 实 际 上 它 也 是 框架 ， 
你 也 许 还 听 到 过 人 们 称 之 为 组 件 库 ， 因 为 我 们 也 可 能 从 一 组 松散 灰 合 的 
组 件 中 挑选 部 分 使 用 ， 而 不 是 要 实现 一 个 结构 化 的 、 架 构 民 好 的 应 用 程 
厅 。 例 如 ，Zend 的 单独 组 件 分 列 解决 了 数据 库 连接 和 存档 、 国 际 化 和 本 
地 化 、 认 证 、 授 权 、 会 话 管 理 、Web 服 务 的 使 用 和 雄 露 、 邮 件 功能 等 等 


诸多 问题 。 


这 个 框架 的 各 个 组 件 根本 没有 任何 负面 的 评价 ， 它 确实 你 证 了 让 我 
们 时 刻 记 得 使 用 框架 的 目标 之 一 ， 束 是 坚持 一 种 易于 维护 的 染 构 模式 ， 
你 的 应 用 程序 将 会 很 整齐 一 致 。 但 如 条 你 的 应 用 程序 包 人 台 了 框架 组 件 的 
一 个 库 ， 并 且 你 只 是 使 用 其 中 的 一 种 ， 你 可 能 要 考虑 一 下 维护 框架 的 过 
度 负 担 “ 更 新 、 打 补丁 等 等 ) 是 合 值得 。 


你 的 答案 可 能 是 有 上 所 不 同 的 ， 但 是 你 应 该 知道 什么 样 的 次 定 更 适合 


目 己 。 要 下 载 和 安装 Zend Framework 以 使 其 与 已 安装 有 的 PHP、Apache 和 
MySQL 一 起 工作 ， 首 先 访 问 位 于 http:/ 
framework.zend.com/manual/en/requirements.introduction.html 的 
Reduirements 页 面 ， 以 确保 你 拥有 PHP 的 正确 版 本 或 者 任何 附加 的 库 和 
模块 。 然 后 ， 访 问 http://framework.zend. com/ download/latest ， 找 到 访 
软件 的 完整 版 或 精简 版 的 链接 。 在 获取 了 你 所 选取 的 格式 的 友 布 文件 
后 ， 只 需要 根据 发 布 版 中 的 说 明 来 提取 它 束 可 以 了 【〔 们 而 言 之 ， 解 压缩 
一 个 目录 并 将 其 放 入 到 已 有 的 文件 系统 中 ) 。Zend Framework REAT] 

Fd bt http://framework.zend.com/ manual 
/en/learning.quickstart.intro.html ， 通 过 示例 应 用 程序 的 创建 ， 同 你 展示 
了 使 用 该 框 染 及 其 组 件 的 整个 过 程 。 


33.3.2 CakePHP 


男 一 个 长 时 间 存 在 的 稳定 的 PHP 应 用 程序 框架 是 
CakePHP(http://www.cakephp.org/ )， 它 也 有 一 个 强大 的 用 户 和 开发 者 社 
区 。 实 际 上 ，CakePHP 是 一 个 MVC 框 六， 其 组 件 具 备 数 据 库 连 接 、 验 
证 、 授 权 、 会 话 管理 、Web 服 务 使 用 和 人 发布 等 章 见 功能 ， 这 些 都 和 Zend 
Framework 相 似 ， 实 际 上 ， 和 很 多 其 他 的 框架 也 是 相似 的 。 


CakePHP 最 大 的 一 个 卖点 是 易于 使 用 《或 者 说 易于 集成 ) ， 而 其 简 
单 性 并 不 影响 到 开 及 和 企业 部 闭 的 丰 军 功能。 和 Zend Framework 一 样 ， 
你 不 必 人 处理 定制 的 配置 文件 ， 只 需要 下 载 和 安 疙 框 染 并 开始 快速 开 友 。 
从 狐 用 户 的 角度 来 看 ，CakePHP 的 优势 在 于 除了 具备 标准 的 开 友 者 论坛 
Lab, “ERA A PEAR AA AC SO, WAR RAE o 


要 下 载 CakePHP 以 配合 已 有 的 PHP、Apache 和 MYySQL 使 用 ， 首 先 访 


la] CakePHP Cookbook 的 安装 页 面 

Chttp://book.cakephp.org/2.0/en/installation.html ) ， 以 确保 你 的 PHP 和 任 
何 附 加 的 库 和 模块 的 版 本 正确 。 然 后 ， 从 链接 进入 下 载 页 面 或 者 站 点 的 
主页 ， 下 载 当 前 的 版 本 。 


TRE SAAR OR Wa. JA m BET Ps iB OK HY Web IRA as HY OCS 
根 目 录 。 在 开 及 环境 和 产品 环境 中 使 用 CakePHP 的 时 候 ， 你 有 几 个 不 同 
的 选项 ， 详 细 介 绍 参阅 Cookbook《〈 也 可 以 访问 位 于 
http://book.cakephp.org/2.0/en/getting-started.html KREA T lisp, LAS 
解 使 用 CakePHP 创 建 一 些 示 例 应 用 程序 的 完整 过 程 。 


33.3.3 Codelgniter 


不 仅 CakePHP 号 称 易 于 使 用 ， 
Codelgniter(http://www.codeigniter.com ) 也 是 初学 者 的 一 个 很 好 的 开端 ， 
其 功能 面 同 发 布 内 容 ， 而 不 是 商务 过 程 或 其 他 的 事务 流程 。 这 并 不 是 什 
么 坏事 情 ， 因 为 在 架构 全 面 的 软件 中 以 一 种 易于 使 用 的 方式 来 友 布 内 
容 ， 这 是 钊 上 添 花 的 事情 。 我 可 能 不 会 在 企业 中 使 用 CodeIgniter， 但 很 
多 Web 开 发 者 由 此 可 以 很 好 地 让 众多 的 客户 感到 满意 ， 而 不 需要 进入 企 
业 空 间 。 因 此 ， 再 次 强调 一 下 ， 要 选择 适合 你 的 框架 。 


Codelgniter 是 一 个 MVC 框 架 ， 此 外 ， 它 还 提供 了 诸 如 数据 库 连 接 、 
验证 、 授 权 、 会 话 官 理 、 本 地 化 、 图 像 操 作 、 使 用 和 发 布 Web 服 务 等 名 
见 的 功能 ， 很 像 是 本 章 中 讨论 的 其 他 的 框架 。 和 本 章 所 介绍 的 其 他 框架 
一 样 ， 其 使 用 过 程 包 括 下 载 和 安装 框架 ， 然 后 束 可 以 立即 用 来 进行 开 
发 。 和 CakePHP 一 样 ，Codelgniter 也 胜 在 除了 具备 标准 的 开发 者 论坛 之 
Sh, IDEA TEA AY ALA ACEP CIS, DR E EA ALTE o 


要 下 载 CodeIgnite 以 配合 已 有 的 PHP、Apache 和 MVySQL 一 起 使 用 ， 
首先 访问 位 于 http:/ codeigniter.com/user_guide/general/requirements.html 
的 Server Requirements 页 面 ， 以 确保 PHP 和 任何 附加 的 库 和 模 顽 的 版 本 
正确 。 然 后 从 位 于 http://codeigniter.com/downloads/ 的 下 载 页 面 下载 正 确 
的 版 本 。 


33.4 小结 


本 章 介 绍 了 使 用 应 用 程序 框架 的 概念 ， 并 且 特 别 介绍 了 使 用 模型 
饮 图 -控制 颖 模式 的 软件 涤 构 。 然 后 ， 介 绍 了 如 何 评估 你 目 己 的 应 用 开 
友 了 折 要 使 用 的 框架 。 最 后 ， 介 绍 了 3 种 常用 的 PHP 应 用 程序 框 染 
(Zend、CakePHP 和 Codelgniter) 。 


33.5 KRAJ 


这 个 实践 练习 设计 用 来 帮助 你 预测 可 能 的 问题 、 复 习 已 经 学 过 的 知 
识 ， 并 且 开 始 把 知识 用 于 实践 。 


1. EH DA REAP NEAR A BE a? 
2. 在 MVC 模 式 中 ， 模 型 做 些 什 么 ? 


3. 你 必须 使 用 应 用 程序 框 淋 吗 ? 


解答 
1. 使 用 一 个 稳定 的 代码 ， 开 从 一 种 软件 架构 模式 ， 并 且 不 用 午 狐 
AHA FEF 
2， 模 型 存储 并 从 控制 和 视图 组 件 分 隔 数据 。 


3. 并 非 如 此 。 实 际 上 ， 本 书 前 面 章 市 所 介绍 的 PHP 和 MySQL 开 发 
的 基本 方面 ， 并 不 依赖 于 任何 框架 。 


EA wel 
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2. 阅读 人 至少 一 种 这 些 框架 的 开发 者 所 提供 的 教程 ， 以 便 获 得 使 用 
框架 和 MVC 模 式 的 一 些 实 用 知识 。 


欢迎 来 到 弄 步 社区 ! 


异步 社区 的 来 历 


异步 社区 (www.epubit.com.cm) 是 人 民 邮 电 出 版 社 旗下 IT 专业 图 书 旗 
舰 社 区 ， 于 2015 年 8 月 上 线 运营 。 


异步 社区 依托 于 人 民 邮 电 出 版 社 20 余 年 的 IT 专业 优质 出 版 资源 和 编 
得 策划 团队 ， 打 造 传统 出 版 与 电子 出 版 和 自 出 版 结合 、 纸 质 书 与 电子 书 
结合 、 传 统 印刷 与 POD 按 需 印刷 结合 的 出 版 平台 ， 提 供 最 新 技术 资讯 ， 
为 作者 和 读者 打造 交流 互动 的 平台 
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免费 电子 书 


Free eBook 
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Write for Us 
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我 们 出 版 的 图 书 涵盖 主流 IT 技术 ， 在 编程 语言 、Web 技 术 、 数 据 科 
学 等 领域 有 众多 经 典 畅 销 图 书 。 社 区 现 已 上 线 图 书 1000 余 种 ， 电 子 书 
400 多 种 ， 部 分 新 书 实现 纸 书 、 电 子 书 同步 出 版 。 我 们 还 会 定期 发 布 新 
书 书 讯 。 


FRI 
社区 内 提供 随 书 附 赠 的 资源 ， 如 书 中 的 案例 或 程序 源 代码 。 


为 外 ， 社 区 还 提供 了 大 量 的 免费 电子 书 ， 只 要 注册 成 为 社区 用 户 就 
可 以 免费 下 载 。 


SVE Aa 


很 多 图 书 的 作 详 者 已 经 入 驻 社区 ， 您 可 以 关注 他 们 ， 盗 询 拉 术 问 
题 ; 可 以 疯 读 不 断 更 新 的 撤 术 文 草 ， 听 作 详 着 和 编辑 畅 获 好 书 表 后 有 趣 
的 故事 : 还 可 以 参与 社区 的 作者 访谈 栏目 ， 同 您 关注 的 作者 提出 采访 题 
H- 


灵活 优惠 的 购书 


您 可 以 方便 地 下 单 购买 纸 质 图 书 或 电子 图 书 ， 纸 质 图 书 耳 接 从 人 民 
邮电 出 版 社 书 库 友 所， 电子 书 近 供 多 种 阅读 格式 。 


对 于 重 磅 新 书 ， 社 区 提供 预 售 和 新 书 首 肥 服务 ， 用 户 可 以 第 一 时 间 
买 到 心仪 的 新 书 。 


用 尸 帐 尸 中 的 积分 可 以 用 于 购书 优 囊 。100 积 分 =1 元 ， 购 关 图 书 
时 ， 在 ”B39 里 项 入 可 使 用 的 积分 数值 ， 即 可 扣 减 相应 金额 。 


特别 优惠 





购买 本 电子 书 的 读者 专 圣 寞 步 社区 优惠 券 。 使 用 方法 : 注册 成 为 社区 用 户 ， 在 下 单 购书 
时 输入 “57AWG ”， 然 后 点 击 “ 使 用 优惠 码 ?， 即 可 孕 受 电子 书 8 折 优 惠 〈 本 优惠 券 只 可 使 用 一 
次 ) 。 





纸 电 图 书 组 合 购 头 


性 区 独家 提供 纸 质 图 书 和 电子 书 组 合 购买 方式 ， 价 格 优惠 ， 一 次 购 
买 ， 多 种 疝 谈 选择。 


Wireshark 网 络 分 析 的 艺术 

作者 : HTA 
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分 类 ; 计算 机 科学 > 安全 与 加 室 > 网 培 安 全 

Wireshark 旺 当前 最 流行 的 网 络 包 分 析 工具 。 它 上 手 菜单 ， RV. oe 
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竹 区 里 还 可 以 做 什么 ? 
提交 勘误 


您 可 以 在 图 书页 面 下 方 提交 甚 误 ， 每 条 勘误 饿 确认 后 可 以 获得 100 
积分 。 热 心 勘误 的 读者 还 有 机 会 参与 书 往 的 审 校 和 翻 详 工作 。 


任 区 提供 基于 Markdown 的 与 作 环境 ， 豆 欢 与 作 的 您 可 以 在 此 一 试 
身手 ， 在 社区 里 分 吾 您 的 技术 心得 和 读书 体会 ， 更 可 以 体验 目 出 版 的 乐 
趣 ， 轻 松 实现 出 版 的 梦想 。 


如 由 成 为 社区 认证 作 详 者 ， 还 可 以 圣 受 异步 社区 所 供 的 作者 专 圣 特 
色 服 务 。 


会 议 活 动 早 知道 


您 可 以 掌握 IT 圈 的 技术 会 议 资讯 ， 更 有 机 会 免费 获 赠 大 会 门 紧 。 
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微 信 订 阅 号 








家 方 微 博 





QQ 和 群 ， 436746675 


社区 网 址 : www.epubit.com.cn 
官方 做 信 : 异步 社区 
Bm: @ 人 邮 和 异步 社区 ，@ 人 民 邮 电 出 版 社 -信息 技术 分 社 


HAA M: contact@epubit.com.cn 


